// Copyright (c) Lawrence Livermore National Security, LLC and other VisIt
// Project developers.  See the top-level LICENSE file for dates and other
// details.  No copyright assignment is required to contribute to VisIt.

#include <stdio.h>
#include <QvisAnnotationWindow.h>
#include <QButtonGroup>
#include <QCheckBox>
#include <QComboBox>
#include <QGroupBox>
#include <QInputDialog>
#include <QLabel>
#include <QLayout>
#include <QListWidget>
#include <QPushButton>
#include <QRadioButton>
#include <QSpinBox>
#include <QTabWidget>
#include <QTimer>
#include <QToolTip>
#include <QWidget>
#include <QColorDialog>

#include <QNarrowLineEdit.h>
#include <QvisAnnotationObjectInterface.h>
#include <QvisAnnotationObjectInterfaceFactory.h>
#include <QvisAxisAttributesWidget.h>
#include <QvisColorButton.h>
#include <QvisDialogLineEdit.h>
#include <QvisFontAttributesWidget.h>
#include <QvisLineWidthWidget.h>
#include <AnnotationAttributes.h>
#include <AnnotationObject.h>
#include <AnnotationObjectList.h>
#include <PlotList.h>
#include <DataNode.h>
#include <ViewerProxy.h>

#include <DebugStream.h>

// ****************************************************************************
// Method: QvisAnnotationWindow::QvisAnnotationWindow
//
// Purpose: 
//   This is the constructor for the QvisAnnotationWindow class.
//
// Arguments:
//   subj    : The AnnotationAttributes object that the window observes.
//   caption : The string that appears in the window decorations.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:45:35 PST 2001
//
// Modifications:
//   Brad Whitlock, Mon Aug 27 17:15:18 PST 2001
//   Initialized the active tab.
//
//   Brad Whitlock, Fri Feb 15 15:34:46 PST 2002
//   Initialized parentless widgets.
//
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Added more control over the axes tick marks and labels.
//
//   Brad Whitlock, Thu Oct 30 17:40:39 PST 2003
//   Changed it to a QvisPostableWindowSimpleObserver and added objButtonGroup.
//
//   Brad Whitlock, Wed Mar 21 21:08:54 PST 2007
//   Made it observe the plot list.
//
//   Brad Whitlock, Thu Feb 7 16:32:02 PST 2008
//   Removed some widgets.
//
//   Brad Whitlock, Wed Apr  9 10:59:06 PDT 2008
//   QString for caption, shortName.
//
// ****************************************************************************

QvisAnnotationWindow::QvisAnnotationWindow(const QString &caption,
    const QString &shortName, QvisNotepadArea *notepad) :
    QvisPostableWindowSimpleObserver(caption, shortName, notepad,
    QvisPostableWindowSimpleObserver::AllExtraButtons, false)
{
    // Initialize the subject pointers.
    annotationAtts = 0;
    annotationObjectList = 0;
    plotList = 0;

    objectInterfaces = 0;
    nObjectInterfaces = 0;
    displayInterface = 0;

    activeTab = 0;

    // Initialize parentless widgets.
    backgroundStyleButtons = 0;

    objButtonGroup = 0;
}

// ****************************************************************************
// Method: QvisAnnotationWindow::~QvisAnnotationWindow
//
// Purpose: 
//   This is the destructor for the QvisAnnotationWindow class.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:46:30 PST 2001
//
// Modifications:
//   Brad Whitlock, Fri Feb 15 15:34:19 PST 2002
//   Deleted parentless widgets.
//   
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Added more control over the axes tick marks and labels.
//
//   Brad Whitlock, Thu Oct 30 16:49:20 PST 2003
//   Added detachment of subjects and deletion of objButtonGroup.
//
//   Brad Whitlock, Wed Mar 21 21:09:21 PST 2007
//   Detach plot list.
//
//   Brad Whitlock, Thu Feb 7 16:32:19 PST 2008
//   Removed some widgets.
//
// ****************************************************************************

QvisAnnotationWindow::~QvisAnnotationWindow()
{
    if(annotationAtts)
        annotationAtts->Detach(this);

    if(annotationObjectList)
        annotationObjectList->Detach(this);

    if(plotList)
        plotList->Detach(this);

    delete [] objectInterfaces;
}

// ****************************************************************************
// Method: QvisAnnotationWindow::ConnectAnnotationAttributes
//
// Purpose: 
//   Makes the window observe the annotation attributes.
//
// Programmer: Brad Whitlock
// Creation:   Fri Oct 31 14:29:36 PST 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::ConnectAnnotationAttributes(AnnotationAttributes *a)
{
    annotationAtts = a;
    a->Attach(this);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::ConnectAnnotationObjectList.
//
// Purpose: 
//   Makes the window observe the annotation object list.
//
// Programmer: Brad Whitlock
// Creation:   Fri Oct 31 14:29:36 PST 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::ConnectAnnotationObjectList(AnnotationObjectList *a)
{
    annotationObjectList = a;
    a->Attach(this);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::ConnectPlotList
//
// Purpose: 
//   Makes the window observe the plot list.
//
// Programmer: Brad Whitlock
// Creation:   Wed Mar 21 21:09:45 PST 2007
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::ConnectPlotList(PlotList *pl)
{
    plotList = pl;
    pl->Attach(this);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::SubjectRemoved
//
// Purpose: 
//   This method disconnects a subject from the window.
//
// Arguments:
//   TheRemovedSubject : The subject to detach.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:16:47 PST 2003
//
// Modifications:
//   Brad Whitlock, Wed Mar 21 21:10:32 PST 2007
//   Added the plot list.
//
// ****************************************************************************

void
QvisAnnotationWindow::SubjectRemoved(Subject *TheRemovedSubject)
{
    if(TheRemovedSubject == annotationAtts)
        annotationAtts = 0;

    if(TheRemovedSubject == annotationObjectList)
        annotationObjectList = 0;

    if(TheRemovedSubject == plotList)
        plotList = 0;
}

// ****************************************************************************
// Method: QvisAnnotationWindow::CreateWindowContents
//
// Purpose: 
//   This method creates all of the widgets for the window.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:46:53 PST 2001
//
// Modifications:
//   Kathleen Bonnell, Fri Jul  6 14:48:53 PDT 2001
//   Enable gridlines for 2D.
//
//   Brad Whitlock, Mon Aug 27 14:37:28 PST 2001
//   Added color buttons to control the background and foreground colors.
//
//   Kathleen Bonnell, Fri Aug  3 15:04:32 PDT 2001 
//   Enable gridlines for 3D.  Added static edges to 3d axes-type options.
//
//   Brad Whitlock, Fri Sep 21 15:53:34 PST 2001
//   Changed the code so the triad and bbox toggles are not part of the
//   page3d groupbox.
//
//   Kathleen Bonnell, Wed Nov  7 17:45:20 PST 2001
//   Added another option to axes3DTypeComboBox. 
//
//   Brad Whitlock, Wed Sep 19 15:40:50 PST 2001
//   I removed an unused layout.
//
//   Brad Whitlock, Thu Jan 10 08:34:41 PDT 2002
//   I added a checkbox to toggle the visibility of the user information.
//
//   Brad Whitlock, Thu Apr 11 11:43:49 PDT 2002
//   I added checkboxes for the legend and the database.
//
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Added more control over the axes tick marks and labels.
//
//   Eric Brugger, Tue Jun 24 16:04:01 PDT 2003
//   Added the ability to control the 2d axes line width and replaced the
//   2d font size setting with individual controls for setting the x label,
//   y label, x title, and y title font heights.
//
//   Brad Whitlock, Mon Nov 10 16:28:36 PST 2003
//   I added a button that can turn off all annotations.
//
//   Brad Whitlock, Thu Oct 30 16:36:49 PST 2003
//   I moved the code that creates the tabbed controls to different methods.
//
//   Cyrus Harrison, Mon Jun 18 08:57:46 PDT 2007
//   Added database info path expansion mode label and combo box.
//
//   Cyrus Harrison, Tue Sep 25 10:44:04 PDT 2007
//   Moved general options to a new tab
//
//   Brad Whitlock, Wed Jun 25 09:30:20 PDT 2008
//   Qt 4.
//
//   Jeremy Meredith, Tue Nov 18 15:45:15 EST 2008
//   Added options for AxisArray modality.
//
// ****************************************************************************

void
QvisAnnotationWindow::CreateWindowContents()
{
    // Create the tab widget.
    tabs = new QTabWidget(central);
    connect(tabs, SIGNAL(currentChanged(int)),
            this, SLOT(tabSelected(int)));

    topLayout->addWidget(tabs);    

    //
    // Create the window's tabs.
    ///
    CreateGeneralTab();
    Create2DTab();
    Create3DTab();
    CreateArrayTab();
    CreateColorTab();
    CreateObjectsTab();

    // Show the appropriate page based on the activeTab setting.
    tabs->blockSignals(true);
    tabs->setCurrentIndex(activeTab);
    tabs->blockSignals(false);
}


// ****************************************************************************
// Method: QvisAnnotationWindow::CreateGeneralTab
//
// Purpose: 
//   Creates the general options tab.
//
// Note:       I moved this code from CreateWindowContents.
//
// Programmer: Cyrus Harrison
// Creation:   Tue Oct  2 09:29:53 PDT 2007
//
// Modifications:
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Wed Jun 25 09:30:38 PDT 2008
//   Qt 4.
//
//   Kathleen Biagas, Wed Sep  7 16:16:36 PDT 2011
//   Add timeInfo.
//
// ****************************************************************************

void
QvisAnnotationWindow::CreateGeneralTab()
{
    //
    // Create the group of widgets that control general annotation options
    //
    pageGeneral = new QWidget(central);
    tabs->addTab(pageGeneral, tr("General"));

    // use two layouts, so we can have a compact look
    QVBoxLayout *glayout = new QVBoxLayout(pageGeneral);
    glayout->setContentsMargins(5,5,5,5);
    QHBoxLayout *hlayout = new QHBoxLayout(0);
    hlayout->setSpacing(5);
    glayout->addLayout(hlayout);
    glayout->addStretch(10);

    // Create a toggle for the legend.
    legendInfo = new QCheckBox(tr("Legend"), pageGeneral);
    connect(legendInfo, SIGNAL(toggled(bool)),
            this, SLOT(legendChecked(bool)));
    hlayout->addWidget(legendInfo);

    // Create a button that can turn off all annotations.
    turnOffAllButton = new QPushButton(tr("No annotations"), pageGeneral);
    connect(turnOffAllButton, SIGNAL(clicked()),
            this, SLOT(turnOffAllAnnotations()));
    hlayout->addWidget(turnOffAllButton);

    //
    // Create the database information
    //
    databaseInfo = new QGroupBox(pageGeneral);
    databaseInfo->setTitle(tr("Database"));
    databaseInfo->setCheckable(true);
    connect(databaseInfo, SIGNAL(toggled(bool)),
            this, SLOT(databaseInfoChecked(bool)));
    glayout->addWidget(databaseInfo);
    QGridLayout *dLayout = new QGridLayout(databaseInfo);
    dLayout->setSpacing(10);
    dLayout->setColumnStretch(1, 10);

    databasePathExpansionMode = new QComboBox(databaseInfo);
    databasePathExpansionMode->addItem(tr("File"));
    databasePathExpansionMode->addItem(tr("Directory"));
    databasePathExpansionMode->addItem(tr("Full"));
    databasePathExpansionMode->addItem(tr("Smart"));
    databasePathExpansionMode->addItem(tr("Smart Directory"));

    connect(databasePathExpansionMode, SIGNAL(activated(int)),
            this, SLOT(databasePathExpansionModeChanged(int)));
    databasePathExpansionModeLabel = new QLabel(tr("Path expansion"), 
        databaseInfo);
    databasePathExpansionModeLabel->setBuddy(databasePathExpansionMode);
    dLayout->addWidget(databasePathExpansionModeLabel, 0, 0);
    dLayout->addWidget(databasePathExpansionMode, 0, 1);

    QFrame *dbSep = new QFrame(databaseInfo);
    dbSep->setFrameStyle(QFrame::HLine + QFrame::Sunken);
    dLayout->addWidget(dbSep, 1, 0, 1, 2);

    databaseInfoFont = new QvisFontAttributesWidget(databaseInfo);
    connect(databaseInfoFont, SIGNAL(fontChanged(const FontAttributes &)),
            this, SLOT(databaseInfoFontChanged(const FontAttributes &)));
    dLayout->addWidget(databaseInfoFont, 2, 0, 1, 2);

    QFrame *dbSep2 = new QFrame(databaseInfo);
    dbSep2->setFrameStyle(QFrame::HLine + QFrame::Sunken);
    dLayout->addWidget(dbSep2, 3, 0, 1, 2);

    timeInfo = new QGroupBox(databaseInfo);
    timeInfo->setTitle(tr("Time"));
    timeInfo->setCheckable(true);
    connect(timeInfo, SIGNAL(toggled(bool)), this, SLOT(timeInfoChecked(bool)));
    dLayout->addWidget(timeInfo, 4, 0, 1, 2);

    QHBoxLayout *htLayout = new QHBoxLayout(timeInfo);
    htLayout->setSpacing(5);
    databaseTimeScale = new QNarrowLineEdit(timeInfo);
    connect(databaseTimeScale, SIGNAL(returnPressed()),
            this, SLOT(databaseTimeScaleChanged()));
    htLayout->addWidget(new QLabel(tr("Time scale factor"), timeInfo));
    htLayout->addWidget(databaseTimeScale);

    databaseTimeOffset = new QNarrowLineEdit(timeInfo);
    connect(databaseTimeOffset, SIGNAL(returnPressed()),
            this, SLOT(databaseTimeOffsetChanged()));
    htLayout->addWidget(new QLabel(tr("Time offset"), timeInfo));
    htLayout->addWidget(databaseTimeOffset);

    //
    // Create the user information
    //
    userInfo = new QGroupBox(pageGeneral);
    userInfo->setTitle(tr("User information"));
    userInfo->setCheckable(true);
    connect(userInfo, SIGNAL(toggled(bool)),
            this, SLOT(userInfoChecked(bool)));
    glayout->addWidget(userInfo);
    QVBoxLayout *uLayout = new QVBoxLayout(userInfo);
    uLayout->setSpacing(5);
    userInfoFont = new QvisFontAttributesWidget(userInfo);
    connect(userInfoFont, SIGNAL(fontChanged(const FontAttributes &)),
            this, SLOT(userInfoFontChanged(const FontAttributes &)));
    uLayout->addWidget(userInfoFont);

    glayout->addStretch(100);
}


// ****************************************************************************
// Method: QvisAnnotationWindow::Create2DTab
//
// Purpose: 
//   Creates the 2D options tab.
//
// Note:       I moved this code from CreateWindowContents.
//
// Programmer: Brad Whitlock
// Creation:   Thu Oct 30 16:35:59 PST 2003
//
// Modifications:
//   Kathleen Bonnell, Tue Dec 16 11:34:33 PST 2003
//   Added button for automatic label scaling, text fields for label exponents.
//
//   Brad Whitlock, Wed Jul 27 16:01:31 PST 2005
//   I made it create a "Grid and Ticks" tab and a "Title and Labels" tab.
//
//   Brad Whitlock, Thu Feb 7 17:01:36 PST 2008
//   I changed the method so it creates "General", "X axis", and "Y axis"
//   tabs instead of the previous organization.
//
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Wed Jun 25 09:48:41 PDT 2008
//   Qt 4.
//
//   Jeremy Meredith, Thu Jan 22 14:53:22 EST 2009
//   Update just the various 2D sub-tabs sensitivity, not the entire
//   2D tab, when "show axes" is unchecked.  (If you set the whole
//   tab, then you've even disabled "show axes" and can't re-check it.)
//
// ****************************************************************************

void
QvisAnnotationWindow::Create2DTab()
{
    //
    // Create the group of 2D-related widgets and add them as a tab.
    //
    page2D = new QWidget(central);
    QVBoxLayout *page2DLayout = new QVBoxLayout(page2D);
    page2DLayout->setSpacing(5);
    page2DLayout->setContentsMargins(10,10,10,10);
    tabs->addTab(page2D, tr("2D"));

    axesFlagToggle2D = new QCheckBox(tr("Show axes"), page2D);
    connect(axesFlagToggle2D, SIGNAL(toggled(bool)),
            this, SLOT(axesFlagChecked2D(bool)));
    page2DLayout->addWidget(axesFlagToggle2D);

    page2DTabs = new QTabWidget(page2D);
    page2DLayout->addWidget(page2DTabs);

    // Create the general options page.
    page2DTabs->addTab(CreateGeneralTab2D(page2DTabs), tr("General"));

    // Add the X-axis page.
    axes2D[0] = new QvisAxisAttributesWidget(page2DTabs, false);
    connect(axes2D[0], SIGNAL(axisChanged(const AxisAttributes &)),
            this, SLOT(xAxisChanged2D(const AxisAttributes &)));
    page2DTabs->addTab(axes2D[0], tr("X axis"));

    // Add the Y-axis page.
    axes2D[1] = new QvisAxisAttributesWidget(page2DTabs, false);
    connect(axes2D[1], SIGNAL(axisChanged(const AxisAttributes &)),
            this, SLOT(yAxisChanged2D(const AxisAttributes &)));
    page2DTabs->addTab(axes2D[1], tr("Y axis"));
}

// ****************************************************************************
// Method: QvisAnnotationWindow::CreateGeneralTab2D
//
// Purpose: 
//   Creates the options for the general 2D tab.
//
// Arguments:
//   parentWidget : The parent of the widgets we'll create.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Thu Feb 7 16:45:23 PST 2008
//
// Modifications:
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Jun 26 10:03:31 PDT 2008
//   Qt 4.
//
// ****************************************************************************

QWidget *
QvisAnnotationWindow::CreateGeneralTab2D(QWidget *parentWidget)
{
    QWidget *top = new QWidget(parentWidget);
    QVBoxLayout *vlayout = new QVBoxLayout(top);
    vlayout->setContentsMargins(10,10,10,10);
    QGridLayout *lLayout = new QGridLayout(0);
    vlayout->addLayout(lLayout);
    vlayout->addStretch(100);
    lLayout->setSpacing(5);
    lLayout->setContentsMargins(0,0,0,0);
    lLayout->setColumnStretch(1, 10);

    // Create auto set scaling check box.
    int row = 0;
    labelAutoSetScalingToggle2D = new QCheckBox(tr("Auto scale label values"),
        top);
    connect(labelAutoSetScalingToggle2D, SIGNAL(toggled(bool)),
            this, SLOT(labelAutoSetScalingChecked2D(bool)));
    lLayout->addWidget(labelAutoSetScalingToggle2D, row, 0, 1, 2);
    ++row;

    // Create auto set ticks check box.
    axesAutoSetTicksToggle2D = new QCheckBox(tr("Auto set ticks"), top);
    connect(axesAutoSetTicksToggle2D, SIGNAL(toggled(bool)),
            this, SLOT(axesAutoSetTicksChecked2D(bool)));
    lLayout->addWidget(axesAutoSetTicksToggle2D, row, 0, 1, 2);
    ++row;

    // Create the 2D tick mark locations combobox.
    axesTickLocationComboBox2D = new QComboBox(top);
    axesTickLocationComboBox2D->addItem(tr("Inside"));
    axesTickLocationComboBox2D->addItem(tr("Outside"));
    axesTickLocationComboBox2D->addItem(tr("Both"));
    connect(axesTickLocationComboBox2D, SIGNAL(activated(int)),
            this, SLOT(axesTickLocationChanged2D(int)));
    lLayout->addWidget(axesTickLocationComboBox2D, row, 1);
    QLabel *l = new QLabel(tr("Tick mark locations"), top);
    l->setBuddy(axesTickLocationComboBox2D);
    lLayout->addWidget(l, row, 0);
    ++row;

    // Create the 2D tick marks combobox.
    axesTicksComboBox2D = new QComboBox(top);
    axesTicksComboBox2D->addItem(tr("Off"));
    axesTicksComboBox2D->addItem(tr("Bottom"));
    axesTicksComboBox2D->addItem(tr("Left"));
    axesTicksComboBox2D->addItem(tr("Bottom-left"));
    axesTicksComboBox2D->addItem(tr("All axes"));
    connect(axesTicksComboBox2D, SIGNAL(activated(int)),
            this, SLOT(axesTicksChanged2D(int)));
    lLayout->addWidget(axesTicksComboBox2D, row, 1);
    l = new QLabel(tr("Show tick marks"), top);
    l->setBuddy(axesTicksComboBox2D);
    lLayout->addWidget(l, row, 0);
    ++row;

    // Create the 2D line width widget.
    axesLineWidth2D = new QvisLineWidthWidget(0, top);
    lLayout->addWidget(axesLineWidth2D, row, 1);
    connect(axesLineWidth2D, SIGNAL(lineWidthChanged(int)),
            this, SLOT(axesLineWidthChanged2D(int)));
    l = new QLabel(tr("Line width"), top);
    l->setBuddy(axesLineWidth2D);
    lLayout->addWidget(l, row, 0);
    ++row;

    return top;
}

// ****************************************************************************
// Method: QvisAnnotationWindow::Create3DTab
//
// Purpose: 
//   Creates the 3D options tab.
//
// Note:       I moved this code from CreateWindowContents.
//
// Programmer: Brad Whitlock
// Creation:   Thu Oct 30 16:34:53 PST 2003
//
// Modifications:
//   Kathleen Bonnell, Tue Dec 16 11:34:33 PST 2003
//   Added button for automatic label scaling, text fields for label exponents.
//
//   Brad Whitlock, Thu Jul 28 09:20:43 PDT 2005
//   I split it into two subtabs and added controls to set the axis titles
//   and units.
//
//   Brad Whitlock, Thu Feb 7 17:03:30 PST 2008
//   I split the tabs into "General", "X axis", "Y axis", and "Z axis".
//
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Jun 26 10:32:05 PDT 2008
//   Qt 4.
//
//   Jeremy Meredith, Thu Jan 22 14:53:22 EST 2009
//   Update just the various 3D sub-tabs sensitivity, not the entire
//   3D tab, when "show axes" is unchecked.  (If you set the whole
//   tab, then you've even disabled "show axes" and can't re-check it.)
//
// ****************************************************************************

void
QvisAnnotationWindow::Create3DTab()
{
    //
    // Create the group of 3D-related widgets.
    //
    page3D = new QWidget(central);
    QVBoxLayout *page3DLayout = new QVBoxLayout(page3D);
    page3DLayout->setSpacing(10);
    page3DLayout->setContentsMargins(10,10,10,10);
    tabs->addTab(page3D, tr("3D"));

    // Create the toggle for drawing the axes.
    QHBoxLayout *buttonLayout = new QHBoxLayout(0);
    page3DLayout->addLayout(buttonLayout);
    axes3DVisible = new QCheckBox(tr("Show axes"), page3D);
    connect(axes3DVisible, SIGNAL(toggled(bool)),
            this, SLOT(axes3DFlagChecked(bool)));
    buttonLayout->addWidget(axes3DVisible);

    // Create the toggle for the triad.
    triadFlagToggle = new QCheckBox(tr("Show triad"), page3D);
    connect(triadFlagToggle, SIGNAL(toggled(bool)),
            this, SLOT(triadFlagChecked(bool)));
    buttonLayout->addWidget(triadFlagToggle);

    // Create the toggle for the bbox.
    bboxFlagToggle = new QCheckBox(tr("Show bounding box"), page3D);
    connect(bboxFlagToggle, SIGNAL(toggled(bool)),
            this, SLOT(bboxFlagChecked(bool)));
    buttonLayout->addWidget(bboxFlagToggle);

    page3DTabs = new QTabWidget(page3D);
    page3DLayout->addWidget(page3DTabs);

    // Create the general 3D options page.
    page3DTabs->addTab(CreateGeneralTab3D(page3DTabs),
        tr("General"));

    // Add the X-axis page.
    axes3D[0] = new QvisAxisAttributesWidget(page3DTabs, true);
    connect(axes3D[0], SIGNAL(axisChanged(const AxisAttributes &)),
            this, SLOT(xAxisChanged(const AxisAttributes &)));
    page3DTabs->addTab(axes3D[0], tr("X axis"));

    // Add the Y-axis page.
    axes3D[1] = new QvisAxisAttributesWidget(page3DTabs, true);
    connect(axes3D[1], SIGNAL(axisChanged(const AxisAttributes &)),
            this, SLOT(yAxisChanged(const AxisAttributes &)));
    page3DTabs->addTab(axes3D[1], tr("Y axis"));

    // Add the Z-axis page.
    axes3D[2] = new QvisAxisAttributesWidget(page3DTabs, true);
    connect(axes3D[2], SIGNAL(axisChanged(const AxisAttributes &)),
            this, SLOT(zAxisChanged(const AxisAttributes &)));
    page3DTabs->addTab(axes3D[2], tr("Z axis"));
}

// ****************************************************************************
// Method: QvisAnnotationWindow::CreateGeneralTab3D
//
// Purpose: 
//   Creates the general 3D options page.
//
// Arguments:
//   parentWidget : The parent widget for the widgets that we're creating.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Thu Feb 7 17:07:22 PST 2008
//
// Modifications:
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Jun 26 10:37:10 PDT 2008
//   Qt 4.
//
//   Hank Childs, Fri May 13 16:00:24 PDT 2011
//   Add widgets for setting bounding box location.
//
//   Kathleen Biagas, Wed Apr  8 07:54:30 PDT 2015
//   Added labels for Tick location, axis type and line width.
//
//   Alister Maguire, Thu Mar  1 16:08:42 PST 2018
//   Added widgets for changing triad attributes. 
//
//   Alister Maguire, Fri Mar  9 10:13:30 PST 2018
//   Added checkbox for setting triad manually. 
//
//   Brad Whitlock, Wed Oct 24 16:05:26 PDT 2018
//   Block some signals to prevent crash with autoupdate when opening 
//   the window.
//
// ****************************************************************************

QWidget *
QvisAnnotationWindow::CreateGeneralTab3D(QWidget *parentWidget)
{
    QWidget *top = new QWidget(parentWidget);
    QVBoxLayout *vlayout = new QVBoxLayout(top);
    vlayout->setContentsMargins(10,10,10,10);
    QGridLayout *rLayout = new QGridLayout(0);
    vlayout->addLayout(rLayout);
    vlayout->addStretch(100);
    rLayout->setSpacing(5);
    rLayout->setContentsMargins(0,0,0,0);
    rLayout->setColumnStretch(1, 10);

    int row = 0;

    // Create the axes group
    axesGroup = new QGroupBox(top);
    axesGroup->setTitle(tr("Axes"));
    rLayout->addWidget(axesGroup, row, 0, 1, 2);
    row++;

    // Create the axes layout
    QGridLayout *axesLayout = new QGridLayout(axesGroup);
    axesLayout->setSpacing(5);
    axesLayout->setContentsMargins(0,0,0,0);
    axesLayout->setColumnStretch(1, 10);
    int axesRow = 0;

    labelAutoSetScalingToggle = new QCheckBox(tr("Auto scale label values"), top);
    connect(labelAutoSetScalingToggle, SIGNAL(toggled(bool)),
            this, SLOT(labelAutoSetScalingChecked(bool)));
    axesLayout->addWidget(labelAutoSetScalingToggle, axesRow, 0, 1, 2);
    ++axesRow;

    // Create auto set ticks check box.
    axesAutoSetTicksToggle = new QCheckBox(tr("Auto set ticks"), top);
    connect(axesAutoSetTicksToggle, SIGNAL(toggled(bool)),
            this, SLOT(axesAutoSetTicksChecked(bool)));
    axesLayout->addWidget(axesAutoSetTicksToggle, axesRow, 0, 1, 2);
    ++axesRow;

    // Create the 3D tick mark locations combobox.
    axes3DTickLocationComboBox = new QComboBox(top);
    axes3DTickLocationComboBox->addItem(tr("Inside"));
    axes3DTickLocationComboBox->addItem(tr("Outside"));
    axes3DTickLocationComboBox->addItem(tr("Both"));
    connect(axes3DTickLocationComboBox, SIGNAL(activated(int)),
            this, SLOT(axes3DTickLocationChanged(int)));
    axesLayout->addWidget(axes3DTickLocationComboBox, axesRow, 1);
    axes3DTickLocationLabel = new QLabel(tr("Tick mark locations"), top);
    axes3DTickLocationLabel->setBuddy(axes3DTickLocationComboBox);
    axesLayout->addWidget(axes3DTickLocationLabel, axesRow, 0);
    ++axesRow;

    // Create the 3D axes type combobox.
    axes3DTypeComboBox = new QComboBox(top);
    axes3DTypeComboBox->addItem(tr("Closest triad"));
    axes3DTypeComboBox->addItem(tr("Furthest triad"));
    axes3DTypeComboBox->addItem(tr("Outside edges"));
    axes3DTypeComboBox->addItem(tr("Static triad"));
    axes3DTypeComboBox->addItem(tr("Static edges"));
    connect(axes3DTypeComboBox, SIGNAL(activated(int)),
            this, SLOT(axes3DTypeChanged(int)));
    axesLayout->addWidget(axes3DTypeComboBox, axesRow, 1);
    axes3DTypeLabel = new QLabel(tr("Axis type"), top);
    axes3DTypeLabel->setBuddy(axes3DTypeComboBox);
    axesLayout->addWidget(axes3DTypeLabel, axesRow, 0);
    ++axesRow;

    // Create the 2D line width widget.
    axesLineWidth = new QvisLineWidthWidget(0, top);
    axesLayout->addWidget(axesLineWidth, axesRow, 1);
    connect(axesLineWidth, SIGNAL(lineWidthChanged(int)),
            this, SLOT(axesLineWidthChanged(int)));
    axesLineWidthLabel = new QLabel(tr("Line width"), top);
    axesLineWidthLabel->setBuddy(axesLineWidth);
    axesLayout->addWidget(axesLineWidthLabel, axesRow, 0);
    ++axesRow;

    // Create the bounding box group
    bboxGroup = new QGroupBox(top);
    bboxGroup->setTitle(tr("Bounding box"));
    rLayout->addWidget(bboxGroup, row, 0, 1, 2);
    row++;

    QGridLayout *bboxLayout = new QGridLayout(bboxGroup);
    bboxLayout->setSpacing(5);
    bboxLayout->setContentsMargins(0,0,0,0);
    bboxLayout->setColumnStretch(1, 10);
    int bboxRow = 0;

    setBBoxLocationToggle = new QCheckBox(tr("Set bounding box location manually"), top);
    connect(setBBoxLocationToggle, SIGNAL(toggled(bool)),
            this, SLOT(setBBoxLocationChecked(bool)));
    bboxLayout->addWidget(setBBoxLocationToggle, bboxRow, 0, 1, 2);
    ++bboxRow;
    
    std::vector<std::string> labels;
    labels.push_back("X minimum");
    labels.push_back("X maximum");
    labels.push_back("Y minimum");
    labels.push_back("Y maximum");
    labels.push_back("Z minimum");
    labels.push_back("Z maximum");

    for (int i = 0 ; i < 6 ; i++)
    {
      bboxLabels[i] = new QLabel(labels[i].c_str());
      bboxLayout->addWidget(bboxLabels[i], bboxRow, 0);
      bboxLocations[i] = new QNarrowLineEdit(top);
      bboxLayout->addWidget(bboxLocations[i], bboxRow, 1);
      ++bboxRow;
      connect(bboxLocations[i], SIGNAL(returnPressed()),
            this, SLOT(bboxLocationChanged()));
    }

    // Create the triad options group
    triadGroup = new QGroupBox(top);
    triadGroup->setTitle(tr("Triad"));
    rLayout->addWidget(triadGroup, row, 0, 1, 2);
    row++;

    QGridLayout *triadLayout = new QGridLayout(triadGroup);
    triadLayout->setSpacing(5);
    triadLayout->setContentsMargins(0,0,0,0);
    triadLayout->setColumnStretch(1, 10);
    int triadRow = 0;

    // Create the triadSetManual checkbox
    triadSetManual = new QCheckBox(tr("Set triad manually"), top);
    connect(triadSetManual, SIGNAL(toggled(bool)),
            this, SLOT(triadSetManualChecked(bool)));
    triadLayout->addWidget(triadSetManual, triadRow, 0);
    ++triadRow;
    
    // Create the triad color picker
    triadColorButton = new QPushButton(tr("Select color"), central);
    connect(triadColorButton, SIGNAL(clicked()), 
            this, SLOT(setTriadColor()));

    triadColorLabel = new QLabel(tr("Color"), top);
    triadColorLabel->setBuddy(triadColorButton);

    triadLayout->addWidget(triadColorLabel, triadRow, 0);
    triadLayout->addWidget(triadColorButton, triadRow, 1);
    triadRow++;


    // Create the triad line width widget
    triadLineWidth = new QvisLineWidthWidget(0, top);
    connect(triadLineWidth, SIGNAL(lineWidthChanged(int)),
            this, SLOT(triadLineWidthChanged(int)));
    triadLineWidthLabel = new QLabel(tr("Line width"), top);
    triadLineWidthLabel->setBuddy(triadLineWidth);

    triadLayout->addWidget(triadLineWidthLabel, triadRow, 0);
    triadLayout->addWidget(triadLineWidth, triadRow, 1);
    triadRow++;
    
    // Create the triad font combo box
    triadFontComboBox = new QComboBox(top);
    triadFontComboBox->addItem(tr("Arial"));
    triadFontComboBox->addItem(tr("Courier"));
    triadFontComboBox->addItem(tr("Times"));
    connect(triadFontComboBox, SIGNAL(activated(int)),
            this, SLOT(triadFontChanged(int)));
    triadFontLabel = new QLabel(tr("Font"), top);
    triadFontLabel->setBuddy(triadFontComboBox);

    triadLayout->addWidget(triadFontLabel, triadRow, 0);
    triadLayout->addWidget(triadFontComboBox, triadRow, 1);
    triadRow++;

    // Create the bold triad check box
    triadBoldToggle = new QCheckBox(tr("Bold"), top);
    connect(triadBoldToggle, SIGNAL(toggled(bool)),
            this, SLOT(triadBoldToggleChecked(bool)));
    triadBoldToggle->blockSignals(true);
    triadBoldToggle->setChecked(
        annotationAtts->GetAxes3D().GetTriadBold());
    triadBoldToggle->blockSignals(false);

    triadLayout->addWidget(triadBoldToggle, triadRow, 0);
    triadRow++;

    // Create the italic triad check box
    triadItalicToggle = new QCheckBox(tr("Italic"), top);
    connect(triadItalicToggle, SIGNAL(toggled(bool)),
            this, SLOT(triadItalicToggleChecked(bool)));
    triadItalicToggle->blockSignals(true);
    triadItalicToggle->setChecked(
        annotationAtts->GetAxes3D().GetTriadItalic());
    triadItalicToggle->blockSignals(false);

    triadLayout->addWidget(triadItalicToggle, triadRow, 0);
    triadRow++;

    return top;
}

// ****************************************************************************
// Method: QvisAnnotationWindow::CreateArrayTab
//
// Purpose: 
//   Creates the AxisArray options tab.
//
// Programmer: Jeremy Meredith
// Creation:   November 18, 2008
//
// Modifications:
//   Brad Whitlock, Tue Nov 18 15:10:02 PST 2008
//   Qt 4.
//
//   Jeremy Meredith, Fri Jan 16 11:14:14 EST 2009
//   Hide the custom title/units and showGrid settings for the axes.
//
// ****************************************************************************

void
QvisAnnotationWindow::CreateArrayTab()
{
    //
    // Create the group of axisarray-related widgets and add them as a tab.
    //
    pageArray = new QWidget(central);
    QVBoxLayout *aLayout = new QVBoxLayout(pageArray);
    aLayout->setSpacing(5);
    aLayout->setContentsMargins(10,10,10,10);
    tabs->addTab(pageArray, tr("Array"));

    axesFlagToggleArray = new QCheckBox(tr("Show axes"), pageArray);
    aLayout->addWidget(axesFlagToggleArray);
    connect(axesFlagToggleArray, SIGNAL(toggled(bool)),
            this, SLOT(axesFlagCheckedArray(bool)));

    axesArrayGroup = new QWidget(pageArray);
    aLayout->addWidget(axesArrayGroup);
    QVBoxLayout *lLayout = new QVBoxLayout(axesArrayGroup);
    lLayout->setSpacing(5);
    QTabWidget *pageArrayTabs = new QTabWidget(axesArrayGroup);
    lLayout->addWidget(pageArrayTabs);

    // Create the general options page.
    pageArrayTabs->addTab(CreateGeneralTabArray(pageArrayTabs), tr("General"));

    // Add the X-axis page.
    axesArray[0] = new QvisAxisAttributesWidget(pageArrayTabs, false, true,
                                                false, false);
    connect(axesArray[0], SIGNAL(axisChanged(const AxisAttributes &)),
            this, SLOT(axisChangedArray(const AxisAttributes &)));
    pageArrayTabs->addTab(axesArray[0], tr("Axes"));
}

// ****************************************************************************
// Method: QvisAnnotationWindow::CreateGeneralTabArray
//
// Purpose: 
//   Creates the options for the general AxisArray tab.
//
// Arguments:
//   parentWidget : The parent of the widgets we'll create.
//
// Returns:    
//
// Note:       
//
// Programmer: Jeremy Meredith
// Creation:   November 18, 2008
//
// Modifications:
//   Brad Whitlock, Tue Nov 18 15:13:09 PST 2008
//   Qt 4.
//
// ****************************************************************************

QWidget *
QvisAnnotationWindow::CreateGeneralTabArray(QWidget *parentWidget)
{
    QWidget *top = new QWidget(parentWidget);
    QVBoxLayout *vlayout = new QVBoxLayout(top);
    vlayout->setContentsMargins(10,10,10,10);
    QGridLayout *lLayout = new QGridLayout(0);
    vlayout->addLayout(lLayout);
    vlayout->addStretch(100);
    lLayout->setSpacing(5);
    lLayout->setContentsMargins(0,0,0,0);
    lLayout->setColumnStretch(1, 10);

    int row = 0;

    // Create ticks visible check box.
    ticksToggleArray = new QCheckBox(tr("Tick marks visible"), top);
    connect(ticksToggleArray, SIGNAL(toggled(bool)),
            this, SLOT(axesTicksChangedArray(bool)));
    lLayout->addWidget(ticksToggleArray, row, 0);
    ++row;

    // Create auto set scaling check box.
    labelAutoSetScalingToggleArray = new QCheckBox(tr("Auto scale label values"), top);
    connect(labelAutoSetScalingToggleArray, SIGNAL(toggled(bool)),
            this, SLOT(labelAutoSetScalingCheckedArray(bool)));
    lLayout->addWidget(labelAutoSetScalingToggleArray, row, 0);
    ++row;

    // Create auto set ticks check box.
    axesAutoSetTicksToggleArray = new QCheckBox(tr("Auto set ticks"), top);
    connect(axesAutoSetTicksToggleArray, SIGNAL(toggled(bool)),
            this, SLOT(axesAutoSetTicksCheckedArray(bool)));
    lLayout->addWidget(axesAutoSetTicksToggleArray, row, 0);
    ++row;

    // Create the Array line width widget.
    axesLineWidthArray = new QvisLineWidthWidget(0, top);
    lLayout->addWidget(axesLineWidthArray, row, 1);
    connect(axesLineWidthArray, SIGNAL(lineWidthChanged(int)),
            this, SLOT(axesLineWidthChangedArray(int)));
    QLabel *l = new QLabel(tr("Line width"), top);
    lLayout->addWidget(l, row, 0);
    ++row;

    return top;
}


// ****************************************************************************
// Method: QvisAnnotationWindow::CreateColorTab
//
// Purpose: 
//   Creates the color tab.
//
// Note:       This code used to be in CreateWindowContents.
//
// Programmer: Brad Whitlock
// Creation:   Thu Oct 30 16:33:39 PST 2003
//
// Modifications:
//   Brad Whitlock, Wed Nov 14 11:34:45 PDT 2007
//   Added background image support.
//
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Jun 26 10:40:10 PDT 2008
//   Qt 4.
//
//   Kathleen Biagas, Wed Jun  8 17:10:30 PDT 2016
//   Set keyboard tracking to false for spin boxes so that 'valueChanged'
//   signal will only emit when 'enter' is pressed or spinbox loses focus.
//
//   Kathleen Biagas, Tue Apr 18 16:34:41 PDT 2023
//   Support Qt6: buttonClicked -> idClicked.
//
// ****************************************************************************

void
QvisAnnotationWindow::CreateColorTab()
{
    int row = 0;

    //
    // Create the group of color-related widgets.
    //
    pageColor = new QWidget(central);
    tabs->addTab(pageColor, tr("Colors"));

    QVBoxLayout *vcLayout = new QVBoxLayout(pageColor);
    vcLayout->setContentsMargins(10,10,10,10);
    QGridLayout *cLayout = new QGridLayout(0);
    cLayout->setSpacing(10);
    vcLayout->addLayout(cLayout);

    // Add the background color widgets.
    backgroundColorButton = new QvisColorButton(pageColor);
    connect(backgroundColorButton, SIGNAL(selectedColor(const QColor &)),
            this, SLOT(backgroundColorChanged(const QColor &)));
    QLabel *bgColorLabel = new QLabel(tr("Background color"), pageColor);
    bgColorLabel->setBuddy(backgroundColorButton);
    cLayout->addWidget(bgColorLabel, row, 0);
    cLayout->addWidget(backgroundColorButton, row, 1, Qt::AlignLeft);
    ++row;

    // Add the foreground color widgets.
    foregroundColorButton = new QvisColorButton(pageColor);
    connect(foregroundColorButton, SIGNAL(selectedColor(const QColor &)),
            this, SLOT(foregroundColorChanged(const QColor &)));
    QLabel *fgColorLabel = new QLabel(tr("Foreground color"), pageColor);
    fgColorLabel->setBuddy(foregroundColorButton);
    cLayout->addWidget(fgColorLabel, row, 0);
    cLayout->addWidget(foregroundColorButton, row, 1, Qt::AlignLeft);
    ++row;

    // Create the background style widgets.
    QLabel *backgroundStyleLabel = new QLabel(tr("Background style"), pageColor);
    cLayout->addWidget(backgroundStyleLabel, row, 0);
    backgroundStyleButtons = new QButtonGroup(pageColor);
#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
    connect(backgroundStyleButtons, SIGNAL(buttonClicked(int)),
            this, SLOT(backgroundStyleChanged(int)));
#else
    connect(backgroundStyleButtons, SIGNAL(idClicked(int)),
            this, SLOT(backgroundStyleChanged(int)));
#endif
    QGridLayout *mLayout = new QGridLayout(0);
    cLayout->addLayout(mLayout, row, 1, 1, 4);
    QRadioButton *solid = new QRadioButton(tr("Solid"), pageColor);
    backgroundStyleButtons->addButton(solid, 0);
    mLayout->addWidget(solid, 0, 0);
    QRadioButton *gradient = new QRadioButton(tr("Gradient"), pageColor);
    backgroundStyleButtons->addButton(gradient, 1);
    mLayout->addWidget(gradient, 0, 1);
    QRadioButton *image = new QRadioButton(tr("Image"), pageColor);
    backgroundStyleButtons->addButton(image, 2);
    mLayout->addWidget(image, 0, 2);
    QRadioButton *imageSphere = new QRadioButton(tr("Image sphere"), pageColor);
    backgroundStyleButtons->addButton(imageSphere, 3);
    mLayout->addWidget(imageSphere, 0, 3);
    ++row;

    QFrame *splitter = new QFrame(pageColor);
    splitter->setFrameStyle(QFrame::HLine + QFrame::Raised);
    cLayout->addWidget(splitter, row, 0, 1, 5);
    ++row;

    // Create the gradient style combobox.
    gradientStyleComboBox = new QComboBox(pageColor);
    gradientStyleComboBox->addItem(tr("Top to bottom"));
    gradientStyleComboBox->addItem(tr("Bottom to top"));
    gradientStyleComboBox->addItem(tr("Left to right"));
    gradientStyleComboBox->addItem(tr("Right to left"));
    gradientStyleComboBox->addItem(tr("Radial"));
    connect(gradientStyleComboBox, SIGNAL(activated(int)),
            this, SLOT(gradientStyleChanged(int)));
    cLayout->addWidget(gradientStyleComboBox, row, 1, 1, 4);
    gradientStyleLabel = new QLabel(tr("Gradient style"), pageColor);
    gradientStyleLabel->setBuddy(gradientStyleComboBox);
    cLayout->addWidget(gradientStyleLabel, row, 0);
    ++row;

    // Add the gradient color1 widgets.
    gradientColor1Button = new QvisColorButton(pageColor);
    connect(gradientColor1Button, SIGNAL(selectedColor(const QColor &)),
            this, SLOT(gradientColor1Changed(const QColor &)));
    gradientColor1Label = new QLabel(tr("Gradient color 1"), pageColor);
    gradientColor1Label->setBuddy(gradientColor1Button);
    cLayout->addWidget(gradientColor1Label, row, 0);
    cLayout->addWidget(gradientColor1Button, row, 1, Qt::AlignLeft);
    ++row;

    // Add the gradiant color2 widgets.
    gradientColor2Button = new QvisColorButton(pageColor);
    connect(gradientColor2Button, SIGNAL(selectedColor(const QColor &)),
            this, SLOT(gradientColor2Changed(const QColor &)));
    gradientColor2Label = new QLabel(tr("Gradient color 2"), pageColor);
    gradientColor2Label->setBuddy(gradientColor2Button);
    cLayout->addWidget(gradientColor2Label, row, 0);
    cLayout->addWidget(gradientColor2Button, row, 1, Qt::AlignLeft);
    ++row;

    QFrame *splitter2 = new QFrame(pageColor);
    splitter2->setFrameStyle(QFrame::HLine + QFrame::Raised);
    cLayout->addWidget(splitter2, row, 0, 1, 5);  
    ++row;

    // Add the image selection widget
    backgroundImage = new QvisDialogLineEdit(pageColor);
    backgroundImage->setDialogMode(QvisDialogLineEdit::ChooseLocalFile);
    connect(backgroundImage, SIGNAL(returnPressed()),
            this, SLOT(backgroundImageChanged()));
    cLayout->addWidget(backgroundImage, row, 1, 1, 4);
    backgroundImageLabel = new QLabel(tr("Background image"), pageColor);
    backgroundImageLabel->setBuddy(backgroundImage);
    cLayout->addWidget(backgroundImageLabel, row, 0);
    QString disclaimer(tr("The local file must be accessible to the "
        "compute engine in order to be used in scalable rendering mode."));
    backgroundImage->setToolTip(disclaimer);
    backgroundImageLabel->setToolTip(disclaimer);
    ++row;

    // Add the image repeat x,y widgets. 
    imageRepeatX = new QSpinBox(pageColor);
    imageRepeatX->setKeyboardTracking(false);
    imageRepeatX->setMinimum(1);
    imageRepeatX->setMaximum(100);
    imageRepeatX->setButtonSymbols(QSpinBox::PlusMinus);
    connect(imageRepeatX, SIGNAL(valueChanged(int)),
            this, SLOT(imageRepeatXChanged(int)));
    cLayout->addWidget(imageRepeatX, row, 1);
    imageRepeatXLabel = new QLabel(tr("Repetitions in X"), pageColor);
    imageRepeatXLabel->setBuddy(imageRepeatX);
    cLayout->addWidget(imageRepeatXLabel, row, 0);
    ++row;

    imageRepeatY = new QSpinBox(pageColor);
    imageRepeatY->setKeyboardTracking(false);
    imageRepeatY->setMinimum(1);
    imageRepeatY->setMaximum(100);
    imageRepeatY->setButtonSymbols(QSpinBox::PlusMinus);
    connect(imageRepeatY, SIGNAL(valueChanged(int)),
            this, SLOT(imageRepeatYChanged(int)));
    cLayout->addWidget(imageRepeatY, row, 1);
    imageRepeatYLabel = new QLabel(tr("Repetitions in Y"), pageColor);
    imageRepeatYLabel->setBuddy(imageRepeatY);
    cLayout->addWidget(imageRepeatYLabel, row, 0);
    ++row;

    vcLayout->addStretch(50);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::CreateObjectsTab
//
// Purpose: 
//   Creates the objects tab.
//
// Programmer: Brad Whitlock
// Creation:   Thu Oct 30 17:00:19 PST 2003
//
// Modifications:
//   Brad Whitlock, Tue Mar 20 15:19:03 PST 2007
//   Made it so some interfaces don't have to provide a new instance button.
//
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Jun 26 11:00:21 PDT 2008
//   Qt 4.
//
//   Kathleen Biagas, Tue Apr 18 16:34:41 PDT 2023
//   Support Qt6: buttonClicked -> idClicked.
//
// ****************************************************************************

void
QvisAnnotationWindow::CreateObjectsTab()
{
    //
    // Create the group of color-related widgets.
    //
    pageObjects = new QWidget(central);
    tabs->addTab(pageObjects, tr("Objects"));

    QVBoxLayout *objTopLayout = new QVBoxLayout(pageObjects);
    objTopLayout->setContentsMargins(10,10,10,10);
    objTopLayout->setSpacing(5);

    QHBoxLayout *hLayout = new QHBoxLayout(0);
    hLayout->setContentsMargins(0,0,0,0);
    objTopLayout->addLayout(hLayout);

    //
    // Create the buttons that let us create new annotation objects.
    //
    QGroupBox *newObjectGroup = new QGroupBox(pageObjects);
    newObjectGroup->setTitle(tr("Create new"));
    hLayout->addWidget(newObjectGroup);

    QVBoxLayout *objButtonLayout = new QVBoxLayout(newObjectGroup);
    objButtonLayout->setContentsMargins(10,10,10,10);
    objButtonLayout->setSpacing(5);
    objButtonGroup = new QButtonGroup(newObjectGroup);
#if QT_VERSION < QT_VERSION_CHECK(6,0,0)
    connect(objButtonGroup, SIGNAL(buttonClicked(int)),
            this, SLOT(addNewAnnotationObject(int)));
#else
    connect(objButtonGroup, SIGNAL(idClicked(int)),
            this, SLOT(addNewAnnotationObject(int)));
#endif

    //
    // Create the annotation object list and controls to do things to them.
    //
    QGridLayout *annotListLayout = new QGridLayout(0);
    annotListLayout->setContentsMargins(0,0,0,0);
    hLayout->addLayout(annotListLayout);
    hLayout->setStretchFactor(annotListLayout, 10);
    annotListLayout->addWidget(new QLabel(tr("Annotation objects"),
        pageObjects), 0, 0, 1, 2);
    annotationListBox = new QListWidget(pageObjects);
    annotationListBox->setMinimumHeight(100);
    annotationListBox->setSelectionMode(QAbstractItemView::ExtendedSelection);
    connect(annotationListBox, SIGNAL(itemSelectionChanged()),
            this, SLOT(setActiveAnnotations()));
    annotListLayout->addWidget(annotationListBox, 1, 0, 1, 2);
    annotListLayout->setRowStretch(1, 50);

    hideShowAnnotationButton = new QPushButton(tr("Hide/Show"));
    connect(hideShowAnnotationButton, SIGNAL(clicked()),
            this, SLOT(hideActiveAnnotations()));
    annotListLayout->addWidget(hideShowAnnotationButton, 2, 0);

    deleteAnnotationButton = new QPushButton(tr("Delete"));
    connect(deleteAnnotationButton, SIGNAL(clicked()),
            this, SLOT(deleteActiveAnnotations()));
    annotListLayout->addWidget(deleteAnnotationButton, 2, 1);

    //
    // Create all of the annotation interface widgets.
    //
    QvisAnnotationObjectInterfaceFactory factory;
    if(factory.GetMaxInterfaces() > 0)
    {
        nObjectInterfaces = factory.GetMaxInterfaces();
        objectInterfaces = new QvisAnnotationObjectInterface *[nObjectInterfaces];

        bool notFirstInterface = false;
        for(int i = 0; i < factory.GetMaxInterfaces(); ++i)
        {
            objectInterfaces[i] = factory.CreateInterface(
                (AnnotationObject::AnnotationType)i, pageObjects);

            if(objectInterfaces[i])
            {
                // Connect a signal from the annotation interface that allows
                // it to tell this window to apply the changes.
                connect(objectInterfaces[i], SIGNAL(applyChanges()),
                        this, SLOT(applyObjectListChanges()));
                // Connect a signal from the annotation interface that allows
                // it to tell this window to ignore updates.
                connect(objectInterfaces[i], SIGNAL(setUpdateForWindow(bool)),
                        this, SLOT(setUpdateForWindow(bool)));

                // Add a new button to create the annotation.
                if(objectInterfaces[i]->AllowInstantiation())
                {
                    QPushButton *btn = new QPushButton(objectInterfaces[i]->GetName(),
                        newObjectGroup);
                    objButtonGroup->addButton(btn, i);
                    objButtonLayout->addWidget(btn);
                }
    
                // Add the annotation interface to the top object layout.
                objTopLayout->addWidget(objectInterfaces[i]);

                // Hide all but the first interface.
                if(notFirstInterface)
                    objectInterfaces[i]->hide();
                notFirstInterface = true;
            }
            else
            {
                debug1 << "QvisAnnotationObjectInterfaceFactory can't create "
                          "an interface for index=" << i << " yet." << endl;
            }
        }
    }

    objButtonLayout->addStretch(10);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::Update
//
// Purpose: 
//   This method is called when a Subject that this window observes it updated.
//
// Arguments:
//   TheChangedSubject : The subject that is causing the update.
//
// Programmer: Brad Whitlock
// Creation:   Thu Oct 30 18:02:56 PST 2003
//
// Modifications:
//   Brad Whitlock, Wed Mar 21 21:11:09 PST 2007
//   Update the annotation object interfaces when the plot list changes.
//
// ****************************************************************************

void
QvisAnnotationWindow::UpdateWindow(bool doAll)
{
    if(annotationAtts == 0 || annotationObjectList == 0)
        return;

    if(SelectedSubject() == annotationAtts || doAll)
        UpdateAnnotationControls(doAll);
    if(SelectedSubject() == annotationObjectList ||
       SelectedSubject() == plotList ||
       doAll)
        UpdateAnnotationObjectControls(doAll);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::UpdateAxesArray
//
// Purpose: 
//   Updates the 2D axis settings in the window.
//
// Arguments:
//
// Returns:    
//
// Note:       
//
// Programmer: Jeremy Meredith
// Creation:   November 18, 2008
//
// Modifications:
//
//     Alister Maguire, Tue Mar  6 11:12:11 PST 2018 
//     Changed the axes enabler to enable/disable the
//     axesGroup. 
//   
// ****************************************************************************

void
QvisAnnotationWindow::UpdateAxesArray()
{
    const AxesArray &axes = annotationAtts->GetAxesArray();

    axesFlagToggleArray->blockSignals(true);
    axesFlagToggleArray->setChecked(axes.GetVisible());
    axesFlagToggleArray->blockSignals(false);
    axesGroup->setEnabled(axes.GetVisible());

    ticksToggleArray->blockSignals(true);
    ticksToggleArray->setChecked(axes.GetTicksVisible());
    ticksToggleArray->blockSignals(false);
 
    axesAutoSetTicksToggleArray->blockSignals(true);
    axesAutoSetTicksToggleArray->setChecked(axes.GetAutoSetTicks());
    axesAutoSetTicksToggleArray->blockSignals(false);

    labelAutoSetScalingToggleArray->blockSignals(true);
    labelAutoSetScalingToggleArray->setChecked(axes.GetAutoSetScaling());
    labelAutoSetScalingToggleArray->blockSignals(false);

    axesLineWidthArray->blockSignals(true);
    axesLineWidthArray->SetLineWidth(axes.GetLineWidth());
    axesLineWidthArray->blockSignals(false);
 
    // Update the controls in the axes.
    axesArray[0]->setAutoScaling(axes.GetAutoSetScaling());
    axesArray[0]->setAutoTickMarks(axes.GetAutoSetTicks());
    axesArray[0]->setAxisAttributes(axes.GetAxes());
}

// ****************************************************************************
// Method: QvisAnnotationWindow::UpdateAxes2D
//
// Purpose: 
//   Updates the 2D axis settings in the window.
//
// Arguments:
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Thu Feb 7 17:25:58 PST 2008
//
// Modifications:
//   Jeremy Meredith, Thu Jan 22 14:53:22 EST 2009
//   Update just the various 2D sub-tabs sensitivity, not the entire
//   2D tab, when "show axes" is unchecked.  (If you set the whole
//   tab, then you've even disabled "show axes" and can't re-check it.)
//   
// ****************************************************************************

void
QvisAnnotationWindow::UpdateAxes2D()
{
    const Axes2D &axes = annotationAtts->GetAxes2D();

    axesFlagToggle2D->blockSignals(true);
    axesFlagToggle2D->setChecked(axes.GetVisible());
    axesFlagToggle2D->blockSignals(false);

    page2DTabs->setEnabled(axes.GetVisible());

    axesAutoSetTicksToggle2D->blockSignals(true);
    axesAutoSetTicksToggle2D->setChecked(axes.GetAutoSetTicks());
    axesAutoSetTicksToggle2D->blockSignals(false);

    labelAutoSetScalingToggle2D->blockSignals(true);
    labelAutoSetScalingToggle2D->setChecked(axes.GetAutoSetScaling());
    labelAutoSetScalingToggle2D->blockSignals(false);

    axesLineWidth2D->blockSignals(true);
    axesLineWidth2D->SetLineWidth(axes.GetLineWidth());
    axesLineWidth2D->blockSignals(false);
 
    axesTickLocationComboBox2D->blockSignals(true);
    axesTickLocationComboBox2D->setCurrentIndex(axes.GetTickLocation());
    axesTickLocationComboBox2D->blockSignals(false);

    axesTicksComboBox2D->blockSignals(true);
    axesTicksComboBox2D->setCurrentIndex(axes.GetTickAxes());
    axesTicksComboBox2D->blockSignals(false);

    // Update the controls in the axes.
    axes2D[0]->setAutoScaling(axes.GetAutoSetScaling());
    axes2D[0]->setAutoTickMarks(axes.GetAutoSetTicks());
    axes2D[0]->setAxisAttributes(axes.GetXAxis());
    axes2D[1]->setAutoScaling(axes.GetAutoSetScaling());
    axes2D[1]->setAutoTickMarks(axes.GetAutoSetTicks());
    axes2D[1]->setAxisAttributes(axes.GetYAxis());
}

// ****************************************************************************
// Method: QvisAnnotationWindow::UpdateAxes3D
//
// Purpose: 
//   Updates the 3D axis settings in the window.
//
// Arguments:
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Thu Feb 7 17:38:52 PST 2008
//
// Modifications:
//   Jeremy Meredith, Thu Jan 22 14:53:22 EST 2009
//   Update just the various 3D sub-tabs sensitivity, not the entire
//   3D tab, when "show axes" is unchecked.  (If you set the whole
//   tab, then you've even disabled "show axes" and can't re-check it.)
//   
//   Hank Childs, Mon May 23 10:31:29 PDT 2011
//   Add support for setting bounding box location.
//
//   Kathleen Biagas, Wed Apr  8 07:55:22 PDT 2015
//   Modified how widgets are enabled:  'Show axes' contols enabled state of
//   autoSetTicks, autoSetScaling, Tick location and axes type.
//   'Show bounding box' controls enabled state of bounding box controls.
//   If either of these are true, the line width control is enabled.
//
//   Alister Maguire, Tue Mar  6 11:12:11 PST 2018 
//   Changed the enablers to enable/disable by group. 
//
//   Alister Maguire, Fri Mar  9 10:13:30 PST 2018
//   Added enabling/disabling of triad options based on
//   the TriadSetManually flag. 
//
// ****************************************************************************

void
QvisAnnotationWindow::UpdateAxes3D()
{
    const Axes3D &axes = annotationAtts->GetAxes3D();

    triadGroup->setEnabled(axes.GetTriadFlag());
    if (axes.GetTriadSetManually())
    {
        triadColorButton->setEnabled(true); 
        triadColorLabel->setEnabled(true);
        triadLineWidth->setEnabled(true);
        triadLineWidthLabel->setEnabled(true);
        triadFontLabel->setEnabled(true);
        triadFontComboBox->setEnabled(true);
        triadBoldToggle->setEnabled(true);
        triadItalicToggle->setEnabled(true);
    }
    else
    {
        triadColorButton->setEnabled(false); 
        triadColorLabel->setEnabled(false);
        triadLineWidth->setEnabled(false);
        triadLineWidthLabel->setEnabled(false);
        triadFontLabel->setEnabled(false);
        triadFontComboBox->setEnabled(false);
        triadBoldToggle->setEnabled(false);
        triadItalicToggle->setEnabled(false);

        // 
        // Return the triad atts to their original state. 
        //
        int triadColor[3] = {0, 0, 0};  
        annotationAtts->GetAxes3D().SetTriadColor(triadColor);
              
        triadLineWidth->blockSignals(true);
        triadLineWidth->SetLineWidth(0);
        triadLineWidth->blockSignals(false);
        annotationAtts->GetAxes3D().SetTriadLineWidth(0.0);
 
        triadFontComboBox->blockSignals(true);
        triadFontComboBox->setCurrentIndex(0);
        triadFontComboBox->blockSignals(false);
        annotationAtts->GetAxes3D().SetTriadFont(0);

        triadBoldToggle->blockSignals(true);
        triadBoldToggle->setChecked(1);
        triadBoldToggle->blockSignals(false);
        annotationAtts->GetAxes3D().SetTriadBold(1);

        triadItalicToggle->blockSignals(true);
        triadItalicToggle->setChecked(1);
        triadItalicToggle->blockSignals(false);
        annotationAtts->GetAxes3D().SetTriadItalic(1);
    }

    axes3DVisible->blockSignals(true);
    axes3DVisible->setChecked(axes.GetVisible());
    axes3DVisible->blockSignals(false);

    axesGroup->setEnabled(axes.GetVisible());

    axesAutoSetTicksToggle->blockSignals(true);
    axesAutoSetTicksToggle->setChecked(axes.GetAutoSetTicks());
    axesAutoSetTicksToggle->blockSignals(false);

    labelAutoSetScalingToggle->blockSignals(true);
    labelAutoSetScalingToggle->setChecked(axes.GetAutoSetScaling());
    labelAutoSetScalingToggle->blockSignals(false);

    axes3DTickLocationComboBox->blockSignals(true);
    axes3DTickLocationComboBox->setCurrentIndex(axes.GetTickLocation());
    axes3DTickLocationComboBox->blockSignals(false);
    axes3DTickLocationComboBox->setEnabled(axes.GetVisible());

    axes3DTypeComboBox->blockSignals(true);
    axes3DTypeComboBox->setCurrentIndex(axes.GetAxesType());
    axes3DTypeComboBox->blockSignals(false);
    axes3DTypeComboBox->setEnabled(axes.GetVisible());

    axesLineWidth->blockSignals(true);
    axesLineWidth->SetLineWidth(axes.GetLineWidth());
    axesLineWidth->blockSignals(false);
    axesLineWidth->setEnabled(axes.GetVisible() || axes.GetBboxFlag());
    axesLineWidthLabel->setEnabled(axes.GetVisible() || axes.GetBboxFlag());

    triadFlagToggle->blockSignals(true);
    triadFlagToggle->setChecked(axes.GetTriadFlag());
    triadFlagToggle->blockSignals(false);

    bboxFlagToggle->blockSignals(true);
    bboxFlagToggle->setChecked(axes.GetBboxFlag());
    bboxFlagToggle->blockSignals(false);

    // Update the controls in the axes.
    axes3D[0]->setAutoScaling(axes.GetAutoSetScaling());
    axes3D[0]->setAutoTickMarks(axes.GetAutoSetTicks());
    axes3D[0]->setAxisAttributes(axes.GetXAxis());
    axes3D[1]->setAutoScaling(axes.GetAutoSetScaling());
    axes3D[1]->setAutoTickMarks(axes.GetAutoSetTicks());
    axes3D[1]->setAxisAttributes(axes.GetYAxis());
    axes3D[2]->setAutoScaling(axes.GetAutoSetScaling());
    axes3D[2]->setAutoTickMarks(axes.GetAutoSetTicks());
    axes3D[2]->setAxisAttributes(axes.GetZAxis());

    bboxGroup->setEnabled(axes.GetBboxFlag());

    setBBoxLocationToggle->setEnabled(axes.GetBboxFlag());
    setBBoxLocationToggle->blockSignals(true);
    setBBoxLocationToggle->setChecked(axes.GetSetBBoxLocation());

    if (axes.GetSetBBoxLocation() && axes.GetBboxFlag())
    {
        for (int i = 0 ; i < 6 ; i++)
        {
            bboxLocations[i]->setEnabled(true);
            bboxLabels[i]->setEnabled(true);
        }
    }
    else
    {
        for (int i = 0 ; i < 6 ; i++)
        {
            bboxLocations[i]->setEnabled(false);
            bboxLabels[i]->setEnabled(false);
        }
    }
    setBBoxLocationToggle->blockSignals(false);
    
    const double *loc = axes.GetBboxLocation();
    for (int i = 0 ; i < 6 ; i++)
    {
        bboxLocations[i]->blockSignals(true);
        QString val;  val.setNum(loc[i]);
        bboxLocations[i]->setText(val);
        bboxLocations[i]->blockSignals(false);
    }
}

// ****************************************************************************
// Method: QvisAnnotationWindow::UpdateAnnotationControls
//
// Purpose: 
//   This method is called when the annotation attributes object that the
//   window observes is changed. It is this method's responsibility to set the
//   state of the window's widgets to match the state of the annotation
//   attributes.
//
// Arguments:
//   doAll : A flag that tells the method to ignore which attributes are
//           selected. All widgets are updated if doAll is true.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:47:23 PST 2001
//
// Modifications:
//   Kathleen Bonnell, Fri Jul  6 14:48:53 PDT 2001
//   Enable gridlines for 2D.
//
//   Brad Whitlock, Mon Aug 27 16:19:59 PST 2001
//   Added code to set the window colors.
//
//   Kathleen Bonnell, Fri Aug  3 15:04:32 PDT 2001 
//   Enable gridlines for 3D.
//
//   Brad Whitlock, Thu Sep 20 15:46:33 PST 2001
//   Changed code to match the new version of the state object.
//
//   Brad Whitlock, Thu Jan 10 08:41:32 PDT 2002
//   Added a toggle for the user info.
//
//   Brad Whitlock, Thu Apr 11 11:47:42 PDT 2002
//   Added a toggle for the database and the legend.
//
//   Eric Brugger, Tue Nov  5 07:53:55 PST 2002
//   Added more control over the axes tick marks and labels.
//
//   Eric Brugger, Fri Jan 24 11:27:09 PST 2003
//   Changed the way that the axes font sizes were set.
//
//   Eric Brugger, Tue Jun 24 16:04:01 PDT 2003
//   Added the ability to control the 2d axes line width and replaced the
//   2d font size setting with individual controls for setting the x label,
//   y label, x title, and y title font heights.
//
//   Kathleen Bonnell, Tue Dec 16 11:34:33 PST 2003 
//   Added the ability to control the 2d & 3d label scaling exponents.
//
//   Jeremy Meredith, Tue Nov 16 11:39:53 PST 2004
//   Replaced simple QString::sprintf's with a setNum because there seems
//   to be a bug causing numbers to be incremented by .00001.  See '5263.
//
//   Brad Whitlock, Wed Jul 27 15:58:54 PST 2005
//   Added support for setting user-specified titles and labels.
//
//   Brad Whitlock, Wed Nov 14 11:34:45 PDT 2007
//   Added background image support.
//
//   Brad Whitlock, Fri Dec 14 16:41:02 PST 2007
//   Made it use ids for case labels.
//
//   Brad Whitlock, Thu Feb 7 17:43:01 PST 2008
//   Updated to new AnnotationAttributes interface.
//
//   Jeremy Meredith, Tue Nov 18 15:45:15 EST 2008
//   Added options for AxisArray modality.
//
//   Brad Whitlock, Mon Mar  2 14:43:04 PST 2009
//   I added time scale and offset.
//
//   Kathleen Biagas, Wed Sep  7 16:16:56 PDT 2011
//   Added timeInfo.
//
// ****************************************************************************

void
QvisAnnotationWindow::UpdateAnnotationControls(bool doAll)
{
    QColor c;
    bool isGradient, vals[4];
    const unsigned char *cptr = 0;

    // Loop through all the attributes and do something for
    // each of them that changed. This function is only responsible
    // for displaying the state values and setting widget sensitivity.
    for(int i = 0; i < annotationAtts->NumAttributes(); ++i)
    {
        if(!doAll)
        {
            if(!annotationAtts->IsSelected(i))
            continue;
        }

        switch(i)
        {
        case AnnotationAttributes::ID_axes2D:
            UpdateAxes2D();
            break;
        case AnnotationAttributes::ID_axes3D:
            UpdateAxes3D();
            break;
        case AnnotationAttributes::ID_axesArray:
            UpdateAxesArray();
            break;
        case AnnotationAttributes::ID_userInfoFlag:
            userInfo->blockSignals(true);
            userInfo->setChecked(annotationAtts->GetUserInfoFlag());
            userInfo->blockSignals(false);
            break;
        case AnnotationAttributes::ID_userInfoFont:
            userInfoFont->setFontAttributes(annotationAtts->GetUserInfoFont());
            break;
        case AnnotationAttributes::ID_databaseInfoFlag:
            databaseInfo->blockSignals(true);
            databaseInfo->setChecked(annotationAtts->GetDatabaseInfoFlag());
            databaseInfo->blockSignals(false);
            break;
        case AnnotationAttributes::ID_timeInfoFlag:
            timeInfo->blockSignals(true);
            timeInfo->setChecked(annotationAtts->GetTimeInfoFlag() && annotationAtts->GetDatabaseInfoFlag());
            timeInfo->blockSignals(false);
            break;
        case AnnotationAttributes::ID_databaseInfoFont:
            databaseInfoFont->setFontAttributes(annotationAtts->GetDatabaseInfoFont());
            break;
        case AnnotationAttributes::ID_databaseInfoExpansionMode:
            databasePathExpansionMode->blockSignals(true);
            databasePathExpansionMode->setCurrentIndex(
                                annotationAtts->GetDatabaseInfoExpansionMode());
            databasePathExpansionMode->blockSignals(false);
            break;
        case AnnotationAttributes::ID_databaseInfoTimeScale:
            databaseTimeScale->setText(
                QString("%1").arg(annotationAtts->GetDatabaseInfoTimeScale()));
            break;
        case AnnotationAttributes::ID_databaseInfoTimeOffset:
            databaseTimeOffset->setText(
                QString("%1").arg(annotationAtts->GetDatabaseInfoTimeOffset()));
            break;
        case AnnotationAttributes::ID_legendInfoFlag:
            legendInfo->blockSignals(true);
            legendInfo->setChecked(annotationAtts->GetLegendInfoFlag());
            legendInfo->blockSignals(false);
            break;
        case AnnotationAttributes::ID_backgroundColor:
            cptr = annotationAtts->GetBackgroundColor().GetColor();
            c = QColor(int(cptr[0]), int(cptr[1]), int(cptr[2]));
            backgroundColorButton->blockSignals(true);
            backgroundColorButton->setButtonColor(c);
            backgroundColorButton->blockSignals(false);
            break;
        case AnnotationAttributes::ID_foregroundColor:
            cptr = annotationAtts->GetForegroundColor().GetColor();
            c = QColor(int(cptr[0]), int(cptr[1]), int(cptr[2]));
            foregroundColorButton->blockSignals(true);
            foregroundColorButton->setButtonColor(c);
            foregroundColorButton->blockSignals(false);
            break;
        case AnnotationAttributes::ID_gradientBackgroundStyle:
            gradientStyleComboBox->blockSignals(true);
            gradientStyleComboBox->setCurrentIndex(annotationAtts->GetGradientBackgroundStyle());
            gradientStyleComboBox->blockSignals(false);
            break;
        case AnnotationAttributes::ID_gradientColor1:
            cptr = annotationAtts->GetGradientColor1().GetColor();
            c = QColor(int(cptr[0]), int(cptr[1]), int(cptr[2]));
            gradientColor1Button->blockSignals(true);
            gradientColor1Button->setButtonColor(c);
            gradientColor1Button->blockSignals(false);
            break;
        case AnnotationAttributes::ID_gradientColor2:
            cptr = annotationAtts->GetGradientColor2().GetColor();
            c = QColor(int(cptr[0]), int(cptr[1]), int(cptr[2]));
            gradientColor2Button->blockSignals(true);
            gradientColor2Button->setButtonColor(c);
            gradientColor2Button->blockSignals(false);
            break;
        case AnnotationAttributes::ID_backgroundMode:
            vals[0] = annotationAtts->GetBackgroundMode()==AnnotationAttributes::Solid;
            vals[1] = annotationAtts->GetBackgroundMode()==AnnotationAttributes::Gradient;
            vals[2] = annotationAtts->GetBackgroundMode()==AnnotationAttributes::Image;
            vals[3] = annotationAtts->GetBackgroundMode()==AnnotationAttributes::ImageSphere;
            SetButtonGroup(backgroundStyleButtons, vals);

            // Set widget sensitivity based on this field.
            isGradient = (annotationAtts->GetBackgroundMode() == 1);
            gradientStyleLabel->setEnabled(isGradient);
            gradientStyleComboBox->setEnabled(isGradient);
            gradientColor1Label->setEnabled(isGradient);
            gradientColor1Button->setEnabled(isGradient);
            gradientColor2Label->setEnabled(isGradient);
            gradientColor2Button->setEnabled(isGradient);
            backgroundImage->setEnabled(vals[2] || vals[3]);
            backgroundImageLabel->setEnabled(vals[2] || vals[3]);
            imageRepeatX->setEnabled(vals[2] || vals[3]);
            imageRepeatXLabel->setEnabled(vals[2] || vals[3]);
            imageRepeatY->setEnabled(vals[2] || vals[3]);
            imageRepeatYLabel->setEnabled(vals[2] || vals[3]);
            break;
        case AnnotationAttributes::ID_backgroundImage:
            backgroundImage->setText(annotationAtts->GetBackgroundImage().c_str());
            break;
        case AnnotationAttributes::ID_imageRepeatX:
            imageRepeatX->blockSignals(true);
            imageRepeatX->setValue(annotationAtts->GetImageRepeatX());
            imageRepeatX->blockSignals(false);
            break;
        case AnnotationAttributes::ID_imageRepeatY:
            imageRepeatY->blockSignals(true);
            imageRepeatY->setValue(annotationAtts->GetImageRepeatY());
            imageRepeatY->blockSignals(false);
            break;
        }
    } // end for
}

// ****************************************************************************
// Method: QvisAnnotationWindow::UpdateAnnotationObjectControls
//
// Purpose: 
//   This method is called when the annotation object list from the viewer is
//   modified.
//
// Programmer: Brad Whitlock
// Creation:   Thu Oct 30 18:01:38 PST 2003
//
// Modifications:
//   Brad Whitlock, Tue Mar 20 15:25:46 PST 2007
//   Set the enabled state for the hide/show and delete buttons.
//
//   Brad Whitlock, Thu Jun 26 13:26:45 PDT 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::UpdateAnnotationObjectControls(bool doAll)
{
    //
    // Add all of the annotation objects to the list box.
    //
    QvisAnnotationObjectInterface *firstInterface = 0;
    int i, firstInterfaceIndex = -1;
    bool hideDeleteEnabled = true;
    annotationListBox->blockSignals(true);
    annotationListBox->clear();
    for(i = 0; i < annotationObjectList->GetNumAnnotations(); ++i)
    {
        const AnnotationObject &annot = annotationObjectList->GetAnnotation(i);
        int annotType = int(annot.GetObjectType());

        if(annotType >= 0 && annotType < nObjectInterfaces)
        {
            if(objectInterfaces[annotType] != 0)
            {
                if(firstInterface == 0 && annot.GetActive())
                {
                    firstInterface = objectInterfaces[annotType];
                    firstInterfaceIndex = i;
                }

                // Let the interface determine the text it should display
                // in the menu.
                QString mText(objectInterfaces[annotType]->GetMenuText(annot));
                annotationListBox->addItem(mText);
                annotationListBox->item(i)->setSelected(annot.GetActive());

                if(annot.GetActive())
                    hideDeleteEnabled &= objectInterfaces[annotType]->AllowInstantiation();
            }
        }
    }
    annotationListBox->blockSignals(false);

    // Set the enabled state for the hide/show and delete buttons.
    hideShowAnnotationButton->setEnabled(hideDeleteEnabled);
    deleteAnnotationButton->setEnabled(hideDeleteEnabled);

    //
    // If no objects were selected, then make sure that we show the first
    // interface.
    //
    if(!firstInterface)
    {
        // Look for the first non NULL interface.
        for(i = 0; i < nObjectInterfaces; ++i)
        {
            if(objectInterfaces[i])
            {
                firstInterface = objectInterfaces[i];
                break;
            }
        }
    }

    //
    // Make sure that the appropriate object interface is showing and that
    // it is updated.
    //
    displayInterface = 0;
    for(i = 0; i < nObjectInterfaces; ++i)
    {
        if(objectInterfaces[i])
        {
            if(objectInterfaces[i] == firstInterface)
            {
                if(firstInterfaceIndex != -1)
                {
                    // We have an active annotation interface that we're
                    // making sure gets shown. Make sure that it has the
                    // right values in the widgets.
                    AnnotationObject &annot =
                        annotationObjectList->GetAnnotation(firstInterfaceIndex);
                    objectInterfaces[i]->Update(&annot);
                    objectInterfaces[i]->setEnabled(true);
                    displayInterface = objectInterfaces[i];

                    // Set the current item in the annotation list box.
                    annotationListBox->blockSignals(true);
                    annotationListBox->setCurrentRow(firstInterfaceIndex);
                    annotationListBox->blockSignals(false);
                }
                else
                    objectInterfaces[i]->setEnabled(false);
                objectInterfaces[i]->show();
            }
            else
                objectInterfaces[i]->hide();
        }
    }
}

// ****************************************************************************
// Method: QvisViewWindow::GetCurrentValues
//
// Purpose:
//   Get the current values for the text fields.
//
// Programmer: Eric Brugger
// Creation:   Mon Nov  4 12:21:02 PST 2002
//
// Modifications:
//   Brad Whitlock, Fri Feb 8 10:47:31 PDT 2008
//   Totally rewrote.
//
//   Jeremy Meredith, Tue Nov 18 15:45:15 EST 2008
//   Added options for AxisArray modality.
//
//   Brad Whitlock, Mon Mar  2 14:40:38 PST 2009
//   I added database time scale and offset.
//
//   Hank Childs, Mon May 23 10:31:29 PDT 2011
//   Added support for setting the bounding box location.
//
//   Kathleen Biagas, Wed Jun  8 17:10:30 PDT 2016
//   Ensure spinbox values are retrieved.
//
//   Kathleen Biagas, Mon Apr  8 15:43:24 PDT 2019
//   Set userInfoFont, not databaseInfoFont when processing userInfoFont.
//
// ****************************************************************************

void
QvisAnnotationWindow::GetCurrentValues(int which_widget)
{
    bool doAll = (which_widget == -1);

    if(which_widget == AnnotationAttributes::ID_axes2D || doAll)
    {
        annotationAtts->GetAxes2D().SetXAxis(axes2D[0]->getAxisAttributes());
        annotationAtts->GetAxes2D().SetYAxis(axes2D[1]->getAxisAttributes());
        annotationAtts->SelectAxes2D();
    }

    if(which_widget == AnnotationAttributes::ID_axes3D || doAll)
    {
        annotationAtts->GetAxes3D().SetXAxis(axes3D[0]->getAxisAttributes());
        annotationAtts->GetAxes3D().SetYAxis(axes3D[1]->getAxisAttributes());
        annotationAtts->GetAxes3D().SetZAxis(axes3D[2]->getAxisAttributes());
        double loc[6];
        for (int i = 0 ; i < 6 ; i++)
        {
            QString temp(bboxLocations[i]->displayText().trimmed());
            loc[i] = temp.toDouble();
        }
        annotationAtts->GetAxes3D().SetBboxLocation(loc);
        annotationAtts->SelectAxes3D();
    }

    if(which_widget == AnnotationAttributes::ID_axesArray || doAll)
    {
        annotationAtts->GetAxesArray().SetAxes(axesArray[0]->getAxisAttributes());
        annotationAtts->SelectAxesArray();
    }

    if(which_widget == AnnotationAttributes::ID_userInfoFont || doAll)
    {
        annotationAtts->SetUserInfoFont(userInfoFont->getFontAttributes());
    }

    if(which_widget == AnnotationAttributes::ID_databaseInfoFont || doAll)
    {
        annotationAtts->SetDatabaseInfoFont(databaseInfoFont->getFontAttributes());
    }

    if (which_widget == AnnotationAttributes::ID_backgroundImage || doAll)
    {
        QString temp(backgroundImage->displayText().trimmed());
        annotationAtts->SetBackgroundImage(temp.toStdString());
    }

    if (which_widget == AnnotationAttributes::ID_databaseInfoTimeScale || doAll)
    {
        QString temp(databaseTimeScale->displayText().trimmed());
        annotationAtts->SetDatabaseInfoTimeScale(temp.toDouble());
    }

    if (which_widget == AnnotationAttributes::ID_databaseInfoTimeOffset || doAll)
    {
        QString temp(databaseTimeOffset->displayText().trimmed());
        annotationAtts->SetDatabaseInfoTimeOffset(temp.toDouble());
    }

    if (which_widget == AnnotationAttributes::ID_imageRepeatX || doAll)
    {
        if (imageRepeatX->value() != annotationAtts->GetImageRepeatX())
            annotationAtts->SetImageRepeatX(imageRepeatX->value());
    }

    if (which_widget == AnnotationAttributes::ID_imageRepeatY || doAll)
    {
        if (imageRepeatY->value() != annotationAtts->GetImageRepeatY())
            annotationAtts->SetImageRepeatY(imageRepeatY->value());
    }
}

// ****************************************************************************
// Method: QvisAnnotationWindow::SetButtonGroup
//
// Purpose: 
//   Sets the toggle state for all checkboxes in a button group.
//
// Arguments:
//   bg   : The button group that we're setting.
//   vals : An array containing the values that are used to set the toggles.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:49:34 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Jun 26 13:28:52 PDT 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::SetButtonGroup(QButtonGroup *bg, bool *vals)
{
    bg->blockSignals(true);
    QList<QAbstractButton *> buttons(bg->buttons());
    for(int i = 0; i < buttons.count(); ++i)
    {
        QCheckBox *cb = (QCheckBox *)buttons[i];
        cb->blockSignals(true);
        cb->setChecked(vals[i]);
        cb->blockSignals(false);
    }
    bg->blockSignals(false);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::Apply
//
// Purpose: 
//   This method tells the annotation attributes to notify all observers of
//   changes in the object.
//
// Arguments:
//   dontIgnore : If this is true, the new state takes effect immediately.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:50:38 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Aug 30 10:46:21 PDT 2001
//   Updated the code to reflect name change to SetAnnotationAttributes.
//
//   Eric Brugger, Tue Nov  5 07:53:55 PST 2002
//   Added more control over the axes tick marks and labels.
//
// ****************************************************************************

void
QvisAnnotationWindow::Apply(bool dontIgnore)
{
    if(AutoUpdate() || dontIgnore)
    {
        // Get the current annotation attributes and tell the other
        // observers about them.
        GetCurrentValues(-1);

        annotationAtts->Notify();
        GetViewerMethods()->SetAnnotationAttributes();
    }
    else
        annotationAtts->Notify();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::ApplyObjectList
//
// Purpose: 
//   This method is called when an annotation interface wants to tell the
//   viewer about a change to the annotation attributes.
//
// Arguments:
//   dontIgnore : If this is true, the new state takes effect immediately.
//
// Programmer: Brad Whitlock
// Creation:   Tue Nov 4 12:27:38 PDT 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::ApplyObjectList(bool dontIgnore)
{
    // If we're showing an interface, make it get its current values.
    if(displayInterface)
        displayInterface->GetCurrentValues(-1);

    annotationObjectList->Notify();

    if(AutoUpdate() || dontIgnore)
    {
        GetViewerMethods()->SetAnnotationObjectOptions();
    }
}

// ****************************************************************************
// Method: QvisPostableWindow::CreateNode
//
// Purpose: 
//   Writes the window's extra information to the config file.
//
// Arguments:
//   parentNode : The node to which the window's attributes are added.
//
// Programmer: Brad Whitlock
// Creation:   Mon Aug 27 17:11:48 PST 2001
//
// Modifications:
//   Brad Whitlock, Fri Oct 31 13:43:39 PST 2003
//   I made it be a QvisPostableWindowSimpleObserver.
//
//   Brad Whitlock, Thu Jun 26 13:30:44 PDT 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::CreateNode(DataNode *parentNode)
{
    // Call the base class's method to save the generic window attributes.
    QvisPostableWindowSimpleObserver::CreateNode(parentNode);

    if(saveWindowDefaults)
    {
        DataNode *node = parentNode->GetNode(windowTitle().toStdString());

        // Save the current tab.
        node->AddNode(new DataNode("activeTab", activeTab));
    }
}

// ****************************************************************************
// Method: QvisAnnotationWindow::SetFromNode
//
// Purpose: 
//   Reads window attributes from the DataNode representation of the config
//   file.
//
// Arguments:
//   parentNode : The data node that contains the window's attributes.
//
// Programmer: Brad Whitlock
// Creation:   Mon Aug 27 17:09:08 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Oct 30 16:41:49 PST 2003
//   I allowed activeTab 3 and made it use QvisPostableWindowSimpleObserver's
//   SetFromNode method.
//
//   Brad Whitlock, Thu Jun 26 13:30:54 PDT 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::SetFromNode(DataNode *parentNode, const int *borders)
{
    DataNode *winNode = parentNode->GetNode(windowTitle().toStdString());
    if(winNode == 0)
        return;

    // Get the active tab and show it.
    DataNode *node;
    if((node = winNode->GetNode("activeTab")) != 0)
    {
        activeTab = node->AsInt();
        if(activeTab < 0 || activeTab > 4)
            activeTab = 0;
    }

    // Call the base class's function.
    QvisPostableWindowSimpleObserver::SetFromNode(parentNode, borders);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::apply
//
// Purpose: 
//   This is a Qt slot function that applies the annotation attributes
//   unconditionally.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:52:03 PST 2001
//
// Modifications:
//   Brad Whitlock, Fri Nov 7 17:46:00 PST 2003
//   I made it apply the object list too.
//
// ****************************************************************************

void
QvisAnnotationWindow::apply()
{
    ApplyObjectList(true);
    Apply(true);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::makeDefault
//
// Purpose: 
//   This is a Qt slot function that is called when the "Make default" button
//   is clicked.
//
// Programmer: Brad Whitlock
// Creation:   Thu Aug 30 09:59:33 PDT 2001
//
// Modifications:
//   Brad Whitlock, Fri Nov 7 17:46:37 PST 2003
//   I added code to tell the viewer to also set the default annotation
//   object list.
//
// ****************************************************************************

void
QvisAnnotationWindow::makeDefault()
{
    // Tell the viewer to set the default annotation attributes.
    annotationAtts->Notify();
    GetViewerMethods()->SetDefaultAnnotationAttributes();

    annotationObjectList->Notify();
    GetViewerMethods()->SetDefaultAnnotationObjectList();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::reset
//
// Purpose: 
//   This is a Qt slot function that is called when the "Reset" button
//   is clicked.
//
// Programmer: Brad Whitlock
// Creation:   Thu Aug 30 09:59:33 PDT 2001
//
// Modifications:
//   Brad Whitlock, Fri Nov 7 17:47:29 PST 2003
//   I added code to tell the viewer to reset the annotation object list.
//
// ****************************************************************************

void
QvisAnnotationWindow::reset()
{
    // Tell the viewer to reset the annotation attributes to the last applied
    // values.
    GetViewerMethods()->ResetAnnotationAttributes();

    // Tell the viewer to reset the annotation object list to the last applied
    // values.
    GetViewerMethods()->ResetAnnotationObjectList();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::tabSelected
//
// Purpose: 
//   This is a Qt slot function that is called when the tabs are changed.
//
// Arguments:
//   index : The new active tab.
//
// Programmer: Brad Whitlock
// Creation:   Mon Aug 27 17:14:17 PST 2001
//
// Modifications:
//   Brad Whitlock, Tue Mar 20 14:33:49 PST 2007
//   Fixed so it uses the new names.
//
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Dec 11 08:47:47 PST 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::tabSelected(int index)
{
    activeTab = index;
}

// ****************************************************************************
// Qt slots for the General tab
// ****************************************************************************

// ****************************************************************************
// Method: QvisAnnotationWindow::userInfoChecked
//
// Purpose: 
//   This is a Qt slot function that sets the user info flag.
//
// Arguments:
//   val : The new user info value.
//
// Programmer: Brad Whitlock
// Creation:   Thu Jan 10 08:43:37 PDT 2002
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::userInfoChecked(bool val)
{
    annotationAtts->SetUserInfoFlag(val);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::userInfoFontChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the user font changes.
//
// Arguments:
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:29:48 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::userInfoFontChanged(const FontAttributes &f)
{
    annotationAtts->SetUserInfoFont(f);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::databaseInfoChecked
//
// Purpose:
//   This is a Qt slot function that sets the database info flag.
//
// Arguments:
//   val : The new database info value.
//
// Programmer: Brad Whitlock
// Creation:   Thu Apr 11 11:49:16 PDT 2002
//
// Modifications:
//   Cyrus Harrison, Mon Jun 18 08:59:15 PDT 2007
//   Added enable/disable for path expansion mode
//
//   Brad Whitlock, Fri Feb 8 13:56:08 PST 2008
//   Moved db expansion coding to the UpdateWindow method.
//
// ****************************************************************************

void
QvisAnnotationWindow::databaseInfoChecked(bool val)
{
    annotationAtts->SetDatabaseInfoFlag(val);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::timeInfoChecked
//
// Purpose:
//   This is a Qt slot function that sets the time info flag.
//
// Arguments:
//   val : The new time info value.
//
// Programmer: Kathleen Biagas 
// Creation:   September 7, 2011
//
// Modifications:
//
// ****************************************************************************

void
QvisAnnotationWindow::timeInfoChecked(bool val)
{
    annotationAtts->SetTimeInfoFlag(val);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::databasePathExpansionModeChanged
//
// Purpose:
//   This is a Qt slot function that sets the database path expansion mode.
//
// Arguments:
//   index : The new path expansion mode
//
// Programmer: Cyrus Harrison
// Creation:   Monday June 18, 2007
//
// Modifications:
//   Cyrus Harrison, Tue Jun 19 09:36:24 PDT 2007
//   Removed cout debug print.
//
//   Cyrus Harrison, Thu Sep 27 09:10:25 PDT 2007
//   Added support for new path expansion options
//
// ****************************************************************************

void
QvisAnnotationWindow::databasePathExpansionModeChanged(int index)
{
    if (index == 0)
    {annotationAtts->SetDatabaseInfoExpansionMode(AnnotationAttributes::File);}
    else if (index == 1)
    {annotationAtts->SetDatabaseInfoExpansionMode(AnnotationAttributes::Directory);}
    else if (index == 2)
    {annotationAtts->SetDatabaseInfoExpansionMode(AnnotationAttributes::Full);}
    else if (index == 3)
    {annotationAtts->SetDatabaseInfoExpansionMode(AnnotationAttributes::Smart);}
    else if (index == 4)
    {annotationAtts->SetDatabaseInfoExpansionMode(AnnotationAttributes::SmartDirectory);}
    
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::databaseInfoFontChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the database font changes.
//
// Arguments:
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:29:48 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::databaseInfoFontChanged(const FontAttributes &f)
{
    annotationAtts->SetDatabaseInfoFont(f);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::bboxLocationChanged
//
// Purpose: 
//   This is a Qt slot that is called when the bbox location changes.
//
// Programmer: Hank Childs
// Creation:   May 23, 2011
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::bboxLocationChanged()
{
    GetCurrentValues(AnnotationAttributes::ID_axes3D);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::triadSetManualChecked
//
// Purpose:
//   This is a Qt slot function that is called when the triad set manual
//   checkbox is toggled. 
//
// Arguments:
//   val:    Whether or not to enable manual override of the triad atts. 
//
// Programmer: Alister Maguire
// Creation:   Fri Mar  9 09:48:42 PST 2018
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::triadSetManualChecked(bool val)
{
    annotationAtts->GetAxes3D().SetTriadSetManually(val);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
//  Method:  QvisAnnotationWindow::setTriadColor
//
//  Purpose:
//   This is a Qt slot function that sets the triad color. 
//
//  Arguments:
//
//  Programmer:  Alister Maguire
//  Creation:    Thu Mar  1 11:11:56 PST 2018
//
// ****************************************************************************
void
QvisAnnotationWindow::setTriadColor()
{
    QColor color = QColorDialog::getColor(Qt::black, this);
    int    iRGB[3];
    iRGB[0] = color.red(); 
    iRGB[1] = color.green(); 
    iRGB[2] = color.blue(); 

    annotationAtts->GetAxes3D().SetTriadColor(iRGB);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::triadLineWidthChanged
//
// Purpose:
//   This is a Qt slot function that is called when the triad line width
//   is changed.
//
// Arguments:
//   index:    The new line width.
//
// Programmer: Alister Maguire
// Creation:   Thu Mar  1 14:43:11 PST 2018
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::triadLineWidthChanged(int index)
{
    annotationAtts->GetAxes3D().SetTriadLineWidth((float)index);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::triadFontChanged
//
// Purpose:
//   This is a Qt slot function that is called when the triad font
//   is changed.
//
// Arguments:
//   index:    The new font.
//
// Programmer: Alister Maguire
// Creation:   Thu Mar  1 14:43:11 PST 2018
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::triadFontChanged(int index)
{
    annotationAtts->GetAxes3D().SetTriadFont(index);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::triadBoldToggleChecked
//
// Purpose:
//   This is a Qt slot function that is called when the triad bold
//   check box is toggled. 
//
// Arguments:
//   val:    Whether or not to enable bold font for the triad. 
//
// Programmer: Alister Maguire
// Creation:   Thu Mar  1 14:43:11 PST 2018
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::triadBoldToggleChecked(bool val)
{
    annotationAtts->GetAxes3D().SetTriadBold(val);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::triadItalicToggleChecked
//
// Purpose:
//   This is a Qt slot function that is called when the triad italic
//   check box is toggled. 
//
// Arguments:
//   val:    Whether or not to enable italic font for the triad. 
//
// Programmer: Alister Maguire
// Creation:   Thu Mar  1 14:43:11 PST 2018
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::triadItalicToggleChecked(bool val)
{
    annotationAtts->GetAxes3D().SetTriadItalic(val);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::databaseTimeScaleChanged
//
// Purpose: 
//   This is a Qt slot that is called when the database time scale changes.
//
// Programmer: Brad Whitlock
// Creation:   Mon Mar  2 14:39:48 PST 2009
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::databaseTimeScaleChanged()
{
    GetCurrentValues(AnnotationAttributes::ID_databaseInfoTimeScale);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::databaseTimeOffsetChanged
//
// Purpose: 
//   This is a Qt slot that is called when the database time offset changes.
//
// Programmer: Brad Whitlock
// Creation:   Mon Mar  2 14:39:48 PST 2009
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::databaseTimeOffsetChanged()
{
    GetCurrentValues(AnnotationAttributes::ID_databaseInfoTimeOffset);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::legendChecked
//
// Purpose: 
//   This is a Qt slot function that sets the legend info flag.
//
// Arguments:
//   val : The new legend info value.
//
// Programmer: Brad Whitlock
// Creation:   Thu Apr 11 11:49:52 PDT 2002
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::legendChecked(bool val)
{
    annotationAtts->SetLegendInfoFlag(val);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::turnOffAllAnnotations
//
// Purpose: 
//   Turns off all annotations.
//
// Programmer: Brad Whitlock
// Creation:   Fri Oct 31 14:40:08 PST 2003
//
// Modifications:
//   Brad Whitlock, Wed Nov 26 14:48:03 PST 2003
//   Added code to hide all of the annotation objects.
//
//   Brad Whitlock, Fri Jan 25 11:50:50 PST 2008
//   Updated for new AnnotationAttributes.
//
// ****************************************************************************

void
QvisAnnotationWindow::turnOffAllAnnotations()
{
    //
    // Hide all of the annotations that are not annotation objects.
    //
    annotationAtts->GetAxes2D().SetVisible(false);
    annotationAtts->GetAxes3D().SetVisible(false);
    annotationAtts->GetAxes3D().SetTriadFlag(false);
    annotationAtts->GetAxes3D().SetBboxFlag(false);
    annotationAtts->SetUserInfoFlag(false);
    annotationAtts->SetDatabaseInfoFlag(false);
    annotationAtts->SetLegendInfoFlag(false);
    Apply();

    //
    // Hide all of the annotation objects.
    //
    if(annotationObjectList->GetNumAnnotations() > 0)
    {
        for(int i = 0; i < annotationObjectList->GetNumAnnotations(); ++i)
        {           
            AnnotationObject &annot = annotationObjectList->operator[](i);
            annot.SetVisible(false);
        }

        ApplyObjectList();
    }
}

// ****************************************************************************
// Qt slots for the 2D tab
// ****************************************************************************

// ****************************************************************************
// Method: QvisAnnotationWindow::axesFlagChecked2D
//
// Purpose: 
//   This is a Qt slot function that sets the 2D flag.
//
// Arguments:
//   val : The new toggle state.
//
// Note:       SetUpdate(false) is not called because we want the widget
//             sensitivity to update.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:52:39 PST 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Renamed the method and modified to match changes in annotationAtts.
//   
//   Brad Whitlock, Fri Feb 8 10:56:51 PDT 2008
//   Updated AnnotationAttributes.
//
// ****************************************************************************

void
QvisAnnotationWindow::axesFlagChecked2D(bool val)
{
    annotationAtts->GetAxes2D().SetVisible(val);
    annotationAtts->SelectAxes2D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axesAutoSetTicksChecked2D
//
// Purpose: 
//   This is a Qt slot function that sets the 2D auto set ticks flag.
//
// Arguments:
//   val : The new toggle state.
//
// Note:       SetUpdate(false) is not called because we want the widget
//             sensitivity to update.
//
// Programmer: Eric Brugger
// Creation:   Mon Nov  4 12:21:02 PST 2002
//
// Modifications:
//   Brad Whitlock, Fri Feb 8 10:56:51 PDT 2008
//   Updated AnnotationAttributes.
//   
// ****************************************************************************

void
QvisAnnotationWindow::axesAutoSetTicksChecked2D(bool val)
{
    annotationAtts->GetAxes2D().SetAutoSetTicks(val);
    annotationAtts->SelectAxes2D();
    Apply();
}


// ****************************************************************************
// Method: QvisAnnotationWindow::labelAutoSetScaling2D
//
// Purpose: 
//   This is a Qt slot function that sets the 2D auto set label scaling flag.
//
// Arguments:
//   val : The new toggle state.
//
// Note:       SetUpdate(false) is not called because we want the widget
//             sensitivity to update.
//
// Programmer: Kathleen Bonnell 
// Creation:   December 11, 2003 
//
// Modifications:
//   Brad Whitlock, Fri Feb 8 10:56:51 PDT 2008
//   Updated AnnotationAttributes.
//   
// ****************************************************************************

void
QvisAnnotationWindow::labelAutoSetScalingChecked2D(bool val)
{
    annotationAtts->GetAxes2D().SetAutoSetScaling(val);
    annotationAtts->SelectAxes2D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::setBBoxLocationChecked
//
// Purpose:
//   This is a Qt slot function that is called when the setBBoxLocation
//   is changed.
//
// Programmer: Hank Childs
// Creation:   May 13, 2011
//
// ****************************************************************************
 
void
QvisAnnotationWindow::setBBoxLocationChecked(bool val)
{
    annotationAtts->GetAxes3D().SetSetBBoxLocation(val);
    annotationAtts->SelectAxes3D();
    SetUpdate(true);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axesLineWidthChanged2D
//
// Purpose:
//   This is a Qt slot function that is called when the 2d axes line width
//   is changed.
//
// Arguments:
//   index:    The new line width.
//
// Programmer: Eric Brugger
// Creation:   Tue Jun 24 16:04:01 PDT 2003
//
// Modifications:
//   Brad Whitlock, Fri Feb 8 10:56:51 PDT 2008
//   Updated AnnotationAttributes.
//
// ****************************************************************************
 
void
QvisAnnotationWindow::axesLineWidthChanged2D(int index)
{
    annotationAtts->GetAxes2D().SetLineWidth(index);
    annotationAtts->SelectAxes2D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axes2DTicksChanged
//
// Purpose: 
//   This is a Qt slot function that tells the attributes which set of 2d
//   tick marks are to be used.
//
// Arguments:
//   index : The new value.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:56:11 PST 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Renamed the method and modified to match changes in annotationAtts.
//
//   Brad Whitlock, Fri Feb 8 11:00:54 PDT 2008
//   Updated AnnotationAttributes.
//
// ****************************************************************************

void
QvisAnnotationWindow::axesTicksChanged2D(int index)
{
    if (index == 0)
        annotationAtts->GetAxes2D().SetTickAxes(Axes2D::Off);
    else if (index == 1)
        annotationAtts->GetAxes2D().SetTickAxes(Axes2D::Bottom);
    else if (index == 2)
        annotationAtts->GetAxes2D().SetTickAxes(Axes2D::Left);
    else if (index == 3)
        annotationAtts->GetAxes2D().SetTickAxes(Axes2D::BottomLeft);
    else if (index == 4)
        annotationAtts->GetAxes2D().SetTickAxes(Axes2D::All);
    annotationAtts->SelectAxes2D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axes2DTickLocationChanged
//
// Purpose: 
//   This is a Qt slot function that tells the attributes where the 2d tick
//   marks will appear.
//
// Arguments:
//   index : The new value.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:56:51 PST 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Renamed the method and modified to match changes in annotationAtts.
//
//   Brad Whitlock, Fri Feb 8 11:03:19 PDT 2008
//   Updated AnnotationAttributes.
//
// ****************************************************************************

void
QvisAnnotationWindow::axesTickLocationChanged2D(int index)
{
    if (index == 0)
        annotationAtts->GetAxes2D().SetTickLocation(Axes2D::Inside);
    else if (index == 1)
        annotationAtts->GetAxes2D().SetTickLocation(Axes2D::Outside);
    else if (index == 2)
        annotationAtts->GetAxes2D().SetTickLocation(Axes2D::Both);
    annotationAtts->SelectAxes2D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::xAxisChanged2D
//
// Purpose: 
//   This is a Qt slot function that is called when anything in the 2D X-axis
//   page changes.
//
// Arguments:
//   aa : The new axis attributes.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:13:19 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::xAxisChanged2D(const AxisAttributes &aa)
{
    annotationAtts->GetAxes2D().SetXAxis(aa);
    annotationAtts->SelectAxes2D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::yAxisChanged2D
//
// Purpose: 
//   This is a Qt slot function that is called when anything in the 2D Y-axis
//   page changes.
//
// Arguments:
//   aa : The new axis attributes.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:13:19 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::yAxisChanged2D(const AxisAttributes &aa)
{
    annotationAtts->GetAxes2D().SetYAxis(aa);
    annotationAtts->SelectAxes2D();
    Apply();
}

// ****************************************************************************
// Qt slots for the 3D tab
// ****************************************************************************

// ****************************************************************************
// Method: QvisAnnotationWindow::axes3DFlagChecked
//
// Purpose: 
//   This is a Qt slot function that sets the 3D flag.
//
// Arguments:
//   val : The new toggle state.
//
// Note:       SetUpdate(false) is not called because we want the widget
//             sensitivity to update.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:52:39 PST 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Modified to match changes in annotationAtts.
//
//   Brad Whitlock, Fri Feb 8 11:04:27 PDT 2008
//   Updated AnnotationAttributes.
//
// ****************************************************************************

void
QvisAnnotationWindow::axes3DFlagChecked(bool val)
{
    annotationAtts->GetAxes3D().SetVisible(val);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::labelAutoSetScaling
//
// Purpose: 
//   This is a Qt slot function that sets the 3D auto set label scaling flag.
//
// Arguments:
//   val : The new toggle state.
//
// Note:       SetUpdate(false) is not called because we want the widget
//             sensitivity to update.
//
// Programmer: Kathleen Bonnell 
// Creation:   December 11, 2003 
//
// Modifications:
//   Brad Whitlock, Fri Feb 8 11:04:27 PDT 2008
//   Updated AnnotationAttributes.
//   
// ****************************************************************************

void
QvisAnnotationWindow::labelAutoSetScalingChecked(bool val)
{
    annotationAtts->GetAxes3D().SetAutoSetScaling(val);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axesAutoSetTicksChecked
//
// Purpose: 
//   This is a Qt slot function that sets the 3D auto set ticks flag.
//
// Arguments:
//   val : The new toggle state.
//
// Note:       SetUpdate(false) is not called because we want the widget
//             sensitivity to update.
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 12:11:47 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::axesAutoSetTicksChecked(bool val)
{
    annotationAtts->GetAxes3D().SetAutoSetTicks(val);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axes3DTickLocationChanged
//
// Purpose: 
//   This is a Qt slot function that tells the attributes where the 3d tick
//   marks will appear.
//
// Arguments:
//   index : The new value.
//
// Programmer: Brad Whitlock
// Creation:   Sun Jun 17 23:59:33 PST 2001
//
// Modifications:
//   Brad Whitlock, Fri Jan 25 11:50:50 PST 2008
//   Updated AnnotationAttributes.
//
// ****************************************************************************

void
QvisAnnotationWindow::axes3DTickLocationChanged(int index)
{
    if (index == 0)
        annotationAtts->GetAxes3D().SetTickLocation(Axes3D::Inside);
    else if (index == 1)
        annotationAtts->GetAxes3D().SetTickLocation(Axes3D::Outside);
    else if (index == 2)
        annotationAtts->GetAxes3D().SetTickLocation(Axes3D::Both);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axes3DTypeChanged
//
// Purpose: 
//   This is a Qt slot function that tells the attributes how the axes will
//   be drawn.
//
// Arguments:
//   index : The new value.
//
// Programmer: Brad Whitlock
// Creation:   Mon Jun 18 00:00:30 PDT 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Modified to match changes in annotationAtts.
//
//   Brad Whitlock, Fri Jan 25 11:50:50 PST 2008
//   Updated AnnotationAttributes.
//   
// ****************************************************************************

void
QvisAnnotationWindow::axes3DTypeChanged(int index)
{
    if (index == 0)
        annotationAtts->GetAxes3D().SetAxesType(Axes3D::ClosestTriad);
    else if (index == 1)
        annotationAtts->GetAxes3D().SetAxesType(Axes3D::FurthestTriad);
    else if (index == 2)
        annotationAtts->GetAxes3D().SetAxesType(Axes3D::OutsideEdges);
    else if (index == 3)
        annotationAtts->GetAxes3D().SetAxesType(Axes3D::StaticTriad);
    else if (index == 4)
        annotationAtts->GetAxes3D().SetAxesType(Axes3D::StaticEdges);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::triadFlagChecked
//
// Purpose: 
//   This is a Qt slot function that tells the attributes whether or not the
//   3D triad should be drawn.
//
// Arguments:
//   val : Whether or not to draw the triad.
//
// Programmer: Brad Whitlock
// Creation:   Mon Jun 18 00:01:11 PDT 2001
//
// Modifications:
//   Brad Whitlock, Fri Jan 25 11:50:50 PST 2008
//   Updated AnnotationAttributes.
//
//   Alister Maguire, Thu Mar  1 16:08:42 PST 2018
//   Removed the SetUpdate so that the triad group
//   will be enabled/disabled. 
//   
// ****************************************************************************

void
QvisAnnotationWindow::triadFlagChecked(bool val)
{
    annotationAtts->GetAxes3D().SetTriadFlag(val);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::bboxFlagChecked
//
// Purpose: 
//   This is a Qt slot function that tells the attributes whether or not to
//   in bbox mode. Is this correct?
//
// Arguments:
//   val : The new value.
//
// Programmer: Brad Whitlock
// Creation:   Mon Jun 18 00:02:06 PDT 2001
//
// Modifications:
//   Brad Whitlock, Fri Jan 25 11:50:50 PST 2008
//   Updated for new AnnotationAttributes.
//   
//   Kathleen Biagas, Wed Apr  8 07:58:02 PDT 2015
//   Allow this toggle to update the window.
//
// ****************************************************************************

void
QvisAnnotationWindow::bboxFlagChecked(bool val)
{
    annotationAtts->GetAxes3D().SetBboxFlag(val);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axesLineWidthChanged
//
// Purpose:
//   This is a Qt slot function that is called when the 3D axes line width
//   is changed.
//
// Arguments:
//   index:    The new line width.
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:16:17 PDT 2008
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::axesLineWidthChanged(int index)
{
    annotationAtts->GetAxes3D().SetLineWidth(index);
    annotationAtts->SelectAxes3D();
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::xAxisChanged
//
// Purpose: 
//   This is a Qt slot function that is called when anything in the 3D X-axis
//   page changes.
//
// Arguments:
//   aa : The new axis attributes.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:13:19 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::xAxisChanged(const AxisAttributes &aa)
{
    annotationAtts->GetAxes3D().SetXAxis(aa);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::yAxisChanged
//
// Purpose: 
//   This is a Qt slot function that is called when anything in the 3D Y-axis
//   page changes.
//
// Arguments:
//   aa : The new axis attributes.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:13:19 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::yAxisChanged(const AxisAttributes &aa)
{
    annotationAtts->GetAxes3D().SetYAxis(aa);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::zAxisChanged
//
// Purpose: 
//   This is a Qt slot function that is called when anything in the 3D Z-axis
//   page changes.
//
// Arguments:
//   aa : The new axis attributes.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Fri Feb 8 11:13:19 PDT 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::zAxisChanged(const AxisAttributes &aa)
{
    annotationAtts->GetAxes3D().SetZAxis(aa);
    annotationAtts->SelectAxes3D();
    Apply();
}

// ****************************************************************************
// Qt slots for the Colors tab
// ****************************************************************************

// ****************************************************************************
// Method: QvisAnnotationWindow::backgroundColorChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the color is changed by
//   using the background color button.
//
// Arguments:
//   c : The new background color.
//
// Programmer: Brad Whitlock
// Creation:   Mon Aug 27 14:47:46 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Sep 20 15:48:24 PST 2001
//   Modified so it conforms to the new state object.
//
// ****************************************************************************

void
QvisAnnotationWindow::backgroundColorChanged(const QColor &c)
{
    ColorAttribute newColor(c.red(), c.green(), c.blue());
    annotationAtts->SetBackgroundColor(newColor);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::foregroundColorChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the color is changed by
//   using the foreground color button.
//
// Arguments:
//   c : The new foreground color.
//
// Programmer: Brad Whitlock
// Creation:   Mon Aug 27 14:47:46 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Sep 20 15:47:51 PST 2001
//   Modified so it conforms to the new state object.
//
// ****************************************************************************

void
QvisAnnotationWindow::foregroundColorChanged(const QColor &c)
{
    ColorAttribute newColor(c.red(), c.green(), c.blue());
    annotationAtts->SetForegroundColor(newColor);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::gradientColor1Changed
//
// Purpose: 
//   This is a Qt slot function that is called when the first gradient color
//   is changed.
//
// Arguments:
//   c : The new color for the first gradient color.
//
// Programmer: Brad Whitlock
// Creation:   Wed Aug 29 17:25:42 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Sep 20 15:49:12 PST 2001
//   Changed the code to conform to the new state object.
//   
// ****************************************************************************

void
QvisAnnotationWindow::gradientColor1Changed(const QColor &c)
{
    ColorAttribute newColor(c.red(), c.green(), c.blue());
    annotationAtts->SetGradientColor1(newColor);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::gradientColor2Changed
//
// Purpose: 
//   This is a Qt slot function that is called when the second gradient color
//   is changed.
//
// Arguments:
//   c : The new color for the second gradient color.
//
// Programmer: Brad Whitlock
// Creation:   Wed Aug 29 17:25:42 PST 2001
//
// Modifications:
//   Brad Whitlock, Thu Sep 20 15:49:12 PST 2001
//   Changed the code to conform to the new state object.
//
// ****************************************************************************

void
QvisAnnotationWindow::gradientColor2Changed(const QColor &c)
{
    ColorAttribute newColor(c.red(), c.green(), c.blue());
    annotationAtts->SetGradientColor2(newColor);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::backgroundStyleChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the user changes the
//   background style radio buttons.
//
// Arguments:
//   index : The new background mode.
//
// Programmer: Brad Whitlock
// Creation:   Wed Aug 29 17:28:26 PST 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Modified to match changes in annotationAtts.
//   
//   Brad Whitlock, Wed Nov 14 13:26:34 PST 2007
//   Added support for image backgrounds.
//
// ****************************************************************************

void
QvisAnnotationWindow::backgroundStyleChanged(int index)
{
    if (index == 0)
        annotationAtts->SetBackgroundMode(AnnotationAttributes::Solid);
    else if (index == 1)
        annotationAtts->SetBackgroundMode(AnnotationAttributes::Gradient);
    else if (index == 2)
        annotationAtts->SetBackgroundMode(AnnotationAttributes::Image);
    else if (index == 3)
        annotationAtts->SetBackgroundMode(AnnotationAttributes::ImageSphere);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::gradientStyleChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the user changes the
//   gradient style radio buttons.
//
// Arguments:
//   index : The new gradient mode.
//
// Programmer: Brad Whitlock
// Creation:   Wed Aug 29 17:28:26 PST 2001
//
// Modifications:
//   Eric Brugger, Mon Nov  4 12:21:02 PST 2002
//   Modified to match changes in annotationAtts.
//   
// ****************************************************************************

void
QvisAnnotationWindow::gradientStyleChanged(int index)
{
    if (index == 0)
        annotationAtts->SetGradientBackgroundStyle(
            AnnotationAttributes::TopToBottom);
    else if (index == 1)
        annotationAtts->SetGradientBackgroundStyle(
            AnnotationAttributes::BottomToTop);
    else if (index == 2)
        annotationAtts->SetGradientBackgroundStyle(
            AnnotationAttributes::LeftToRight);
    else if (index == 3)
        annotationAtts->SetGradientBackgroundStyle(
            AnnotationAttributes::RightToLeft);
    else if (index == 4)
        annotationAtts->SetGradientBackgroundStyle(
            AnnotationAttributes::Radial);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::backgroundImageChanged.
//
// Purpose: 
//   This is a Qt slot function that is called when the background image changes.
//
// Programmer: Brad Whitlock
// Creation:   
//
// Modifications:
//
// ****************************************************************************

void
QvisAnnotationWindow::backgroundImageChanged()
{
    GetCurrentValues(AnnotationAttributes::ID_backgroundImage);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::imageRepeatXChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the number of image repeats
//   in X changes.
//
// Arguments:
//   value : The new number of repeats.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Mon Nov 19 12:11:15 PST 2007
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::imageRepeatXChanged(int value)
{
    annotationAtts->SetImageRepeatX(value);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::imageRepeatYChanged
//
// Purpose: 
//   This is a Qt slot function that is called when the number of image repeats
//   in Y changes.
//
// Arguments:
//   value : The new number of repeats.
//
// Returns:    
//
// Note:       
//
// Programmer: Brad Whitlock
// Creation:   Mon Nov 19 12:11:15 PST 2007
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::imageRepeatYChanged(int value)
{
    annotationAtts->SetImageRepeatY(value);
    SetUpdate(false);
    Apply();
}

// ****************************************************************************
// Qt slots for the Objects tab
// ****************************************************************************

// ****************************************************************************
// Method: QvisAnnotationWindow::setUpdateForWindow
//
// Purpose: 
//   This is a Qt slot function that sets the window's update value, which
//   determines whether or not the window updates when it gets a Notify.
//
// Arguments:
//   val : The new update value.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:11:27 PST 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::setUpdateForWindow(bool val)
{
    SetUpdate(val);
}

// ****************************************************************************
// Method: QvisAnnotationWindow::applyObjectListChanges
//
// Purpose: 
//   This is a Qt slot function that allows the annotation interfaces to
//   force changes to the annotation object list to be sent to the viewer.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:12:14 PST 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::applyObjectListChanges()
{
    ApplyObjectList();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::addNewAnnotationObject
//
// Purpose: 
//   This is a Qt slot function that tells the viewer to create an object of
//   the specified type.
//
// Arguments:
//   annotType : The type of annotation object to create.
//
// Note:       The annotation object list is applied first so we don't lose
//             changes to objects when we create a new one.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:12:54 PST 2003
//
// Modifications:
//   Brad Whitlock, Tue Mar 20 09:47:08 PDT 2007
//   Allow the user to name the object.
//
//   Brad Whitlock, Tue Apr  8 09:27:26 PDT 2008
//   Support for internationalization.
//
//   Brad Whitlock, Thu Jun 26 13:32:21 PDT 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::addNewAnnotationObject(int annotType)
{
    //
    // Make the viewer apply the changes or they will be lost when it creates
    // the new annotation object. This isn't so bad.
    //
    ApplyObjectList(true);

    // Prompt the user for a name for the new annotation object.
    bool ok = false;
    QString newName(GetViewerState()->GetAnnotationObjectList()->GetNewObjectName().c_str());
    QString annotName = QInputDialog::getText(this, "VisIt", 
        tr("Enter a name for the new annotation object."),
        QLineEdit::Normal, newName, &ok);

    //
    // Tell the viewer to create a new annotation object.
    //
    if(ok && !annotName.isEmpty())
    {
        GetViewerMethods()->AddAnnotationObject(annotType, annotName.toStdString());
    }
    else
        Warning(tr("No annotation object was created."));
}

// ****************************************************************************
// Method: QvisAnnotationWindow::setActiveAnnotations
//
// Purpose: 
//   This is a Qt slot function that sets the active annotations.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:14:12 PST 2003
//
// Modifications:
//   Brad Whitlock, Tue Mar 20 15:14:24 PST 2007
//   Disable the hide/show and delete buttons when the interface does not
//   permit objects to be instantiated.
//
//   Brad Whitlock, Thu Jun 26 13:33:04 PDT 2008
//   Qt 4.
//
// ****************************************************************************

void
QvisAnnotationWindow::setActiveAnnotations()
{
    // Set the active flag on the annotation objects in the annotation
    // object list.
    for(int i = 0; i < annotationObjectList->GetNumAnnotations(); ++i)
    {
        bool isSelected = (i < annotationListBox->count()) ?
                          annotationListBox->item(i)->isSelected() : false;
        AnnotationObject &annot = annotationObjectList->operator[](i);
        annot.SetActive(isSelected);
    }
 
    // Apply the changes but make sure that we do it once we're back in
    // the main event loop because updating the annotation object list
    // can clear out and repopulate the annotation list box.
    QTimer::singleShot(100, this, SLOT(applyObjectListChanges()));
}

// ****************************************************************************
// Method: QvisAnnotationWindow::hideActiveAnnotations
//
// Purpose: 
//   This is a Qt slot function that tells the viewer to hide the active
//   annotations.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:14:39 PST 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::hideActiveAnnotations()
{
    //
    // Store any changes made to the widgets since the last apply into the
    // annotation list or else they'll be lost when the viewer sends back
    // the updated annotation list.
    //
    ApplyObjectList(true);

    // Tell the viewer to hide the active annotations.
    GetViewerMethods()->HideActiveAnnotationObjects();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::deleteActiveAnnotations
//
// Purpose: 
//   This is a Qt slot function that tells the viewer to delete the active
//   annotations.
//
// Programmer: Brad Whitlock
// Creation:   Tue Dec 2 15:15:14 PST 2003
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::deleteActiveAnnotations()
{
    //
    // Make the viewer apply any changes or else they'll be lost when the
    // viewer sends back the updated annotation list.
    //
    ApplyObjectList(true);

    // Tell the viewer to delete the active annotations.
    GetViewerMethods()->DeleteActiveAnnotationObjects();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axesLineWidthChangedArray
//
// Purpose:
//   This is a Qt slot function that is called when the axesarray line width
//   is changed.
//
// Arguments:
//   index:    The new line width.
//
// Programmer: Jeremy Meredith
// Creation:   November 18, 2008
//
// Modifications:
//
// ****************************************************************************
 
void
QvisAnnotationWindow::axesLineWidthChangedArray(int index)
{
    annotationAtts->GetAxesArray().SetLineWidth(index);
    annotationAtts->SelectAxesArray();
    SetUpdate(false);
    Apply();
}


// ****************************************************************************
// Method: QvisAnnotationWindow::axisChangedArray
//
// Purpose: 
//   This is a Qt slot function that is called when anything in the axisarray
//   page changes.
//
// Arguments:
//   aa : The new axis attributes.
//
// Returns:    
//
// Note:       
//
// Programmer: Jeremy Meredith
// Creation:   November 18, 2008
//
// Modifications:
//   
// ****************************************************************************

void
QvisAnnotationWindow::axisChangedArray(const AxisAttributes &aa)
{
    annotationAtts->GetAxesArray().SetAxes(aa);
    annotationAtts->SelectAxesArray();
    Apply();
}

// ****************************************************************************
// Method: QvisAnnotationWindow::axesFlagCheckedArray
//
// Purpose: 
//   This is a Qt slot function that sets the axis array visibility
//
// Programmer: Jeremy Meredith
// Creation:   November 18, 2008
//
// Modifications:
//
// ****************************************************************************

void
QvisAnnotationWindow::axesFlagCheckedArray(bool val)
{
    annotationAtts->GetAxesArray().SetVisible(val);
    annotationAtts->SelectAxesArray();
    Apply();
}

// ****************************************************************************
//  Method:  QvisAnnotationWindow::axesTicksChangedArray
//
//  Purpose:
//    Callback when the axisarray tick visibility changes.
//
//  Arguments:
//    
//
//  Programmer:  Jeremy Meredith
//  Creation:    November 18, 2008
//
// ****************************************************************************
void
QvisAnnotationWindow::axesTicksChangedArray(bool val)
{
    annotationAtts->GetAxesArray().SetTicksVisible(val);
    annotationAtts->SelectAxesArray();
    Apply();
}

// ****************************************************************************
//  Method:  QvisAnnotationWindow::labelAutoSetScalingCheckedArray
//
//  Purpose:
//    Callback when the axisarray label autoscaling changes.
//
//  Arguments:
//    
//
//  Programmer:  Jeremy Meredith
//  Creation:    November 18, 2008
//
// ****************************************************************************
void
QvisAnnotationWindow::labelAutoSetScalingCheckedArray(bool val)
{
    annotationAtts->GetAxesArray().SetAutoSetScaling(val);
    annotationAtts->SelectAxesArray();
    Apply();
}

// ****************************************************************************
//  Method:  QvisAnnotationWindow::axesAutoSetTicksCheckedArray
//
//  Purpose:
//    Callback when the axisarray tick autosetting changes.
//
//  Arguments:
//    
//
//  Programmer:  Jeremy Meredith
//  Creation:    November 18, 2008
//
// ****************************************************************************
void
QvisAnnotationWindow::axesAutoSetTicksCheckedArray(bool val)
{
    annotationAtts->GetAxesArray().SetAutoSetTicks(val);
    annotationAtts->SelectAxesArray();
    Apply();
}
